package org.hl7.fhir.r4b.model;

/*
  Copyright (c) 2011+, HL7, Inc.
  All rights reserved.
  
  Redistribution and use in source and binary forms, with or without modification, \
  are permitted provided that the following conditions are met:
  
   * Redistributions of source code must retain the above copyright notice, this \
     list of conditions and the following disclaimer.
   * Redistributions in binary form must reproduce the above copyright notice, \
     this list of conditions and the following disclaimer in the documentation \
     and/or other materials provided with the distribution.
   * Neither the name of HL7 nor the names of its contributors may be used to 
     endorse or promote products derived from this software without specific 
     prior written permission.
  
  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS \"AS IS\" AND \
  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED \
  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. \
  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, \
  INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT \
  NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR \
  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, \
  WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) \
  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE \
  POSSIBILITY OF SUCH DAMAGE.
  */

// Generated on Mon, Jun 13, 2022 17:19+0300 for FHIR v4.3.0

import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.math.*;
import org.hl7.fhir.utilities.Utilities;
import org.hl7.fhir.r4b.model.Enumerations.*;
import org.hl7.fhir.instance.model.api.IBaseBackboneElement;
import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.instance.model.api.ICompositeType;
import ca.uhn.fhir.model.api.annotation.ResourceDef;
import ca.uhn.fhir.model.api.annotation.SearchParamDefinition;
import org.hl7.fhir.instance.model.api.IBaseBackboneElement;
import ca.uhn.fhir.model.api.annotation.Child;
import ca.uhn.fhir.model.api.annotation.ChildOrder;
import ca.uhn.fhir.model.api.annotation.Description;
import ca.uhn.fhir.model.api.annotation.Block;

/**
 * This resource provides: the claim details; adjudication details from the
 * processing of a Claim; and optionally account balance information, for
 * informing the subscriber of the benefits provided.
 */
@ResourceDef(name = "ExplanationOfBenefit", profile = "http://hl7.org/fhir/StructureDefinition/ExplanationOfBenefit")
public class ExplanationOfBenefit extends DomainResource {

  public enum ExplanationOfBenefitStatus {
    /**
     * The resource instance is currently in-force.
     */
    ACTIVE,
    /**
     * The resource instance is withdrawn, rescinded or reversed.
     */
    CANCELLED,
    /**
     * A new resource instance the contents of which is not complete.
     */
    DRAFT,
    /**
     * The resource instance was entered in error.
     */
    ENTEREDINERROR,
    /**
     * added to help the parsers with the generic types
     */
    NULL;

    public static ExplanationOfBenefitStatus fromCode(String codeString) throws FHIRException {
      if (codeString == null || "".equals(codeString))
        return null;
      if ("active".equals(codeString))
        return ACTIVE;
      if ("cancelled".equals(codeString))
        return CANCELLED;
      if ("draft".equals(codeString))
        return DRAFT;
      if ("entered-in-error".equals(codeString))
        return ENTEREDINERROR;
      if (Configuration.isAcceptInvalidEnums())
        return null;
      else
        throw new FHIRException("Unknown ExplanationOfBenefitStatus code '" + codeString + "'");
    }

    public String toCode() {
      switch (this) {
      case ACTIVE:
        return "active";
      case CANCELLED:
        return "cancelled";
      case DRAFT:
        return "draft";
      case ENTEREDINERROR:
        return "entered-in-error";
      case NULL:
        return null;
      default:
        return "?";
      }
    }

    public String getSystem() {
      switch (this) {
      case ACTIVE:
        return "http://hl7.org/fhir/explanationofbenefit-status";
      case CANCELLED:
        return "http://hl7.org/fhir/explanationofbenefit-status";
      case DRAFT:
        return "http://hl7.org/fhir/explanationofbenefit-status";
      case ENTEREDINERROR:
        return "http://hl7.org/fhir/explanationofbenefit-status";
      case NULL:
        return null;
      default:
        return "?";
      }
    }

    public String getDefinition() {
      switch (this) {
      case ACTIVE:
        return "The resource instance is currently in-force.";
      case CANCELLED:
        return "The resource instance is withdrawn, rescinded or reversed.";
      case DRAFT:
        return "A new resource instance the contents of which is not complete.";
      case ENTEREDINERROR:
        return "The resource instance was entered in error.";
      case NULL:
        return null;
      default:
        return "?";
      }
    }

    public String getDisplay() {
      switch (this) {
      case ACTIVE:
        return "Active";
      case CANCELLED:
        return "Cancelled";
      case DRAFT:
        return "Draft";
      case ENTEREDINERROR:
        return "Entered In Error";
      case NULL:
        return null;
      default:
        return "?";
      }
    }
  }

  public static class ExplanationOfBenefitStatusEnumFactory implements EnumFactory<ExplanationOfBenefitStatus> {
    public ExplanationOfBenefitStatus fromCode(String codeString) throws IllegalArgumentException {
      if (codeString == null || "".equals(codeString))
        if (codeString == null || "".equals(codeString))
          return null;
      if ("active".equals(codeString))
        return ExplanationOfBenefitStatus.ACTIVE;
      if ("cancelled".equals(codeString))
        return ExplanationOfBenefitStatus.CANCELLED;
      if ("draft".equals(codeString))
        return ExplanationOfBenefitStatus.DRAFT;
      if ("entered-in-error".equals(codeString))
        return ExplanationOfBenefitStatus.ENTEREDINERROR;
      throw new IllegalArgumentException("Unknown ExplanationOfBenefitStatus code '" + codeString + "'");
    }

    public Enumeration<ExplanationOfBenefitStatus> fromType(PrimitiveType<?> code) throws FHIRException {
      if (code == null)
        return null;
      if (code.isEmpty())
        return new Enumeration<ExplanationOfBenefitStatus>(this, ExplanationOfBenefitStatus.NULL, code);
      String codeString = code.asStringValue();
      if (codeString == null || "".equals(codeString))
        return new Enumeration<ExplanationOfBenefitStatus>(this, ExplanationOfBenefitStatus.NULL, code);
      if ("active".equals(codeString))
        return new Enumeration<ExplanationOfBenefitStatus>(this, ExplanationOfBenefitStatus.ACTIVE, code);
      if ("cancelled".equals(codeString))
        return new Enumeration<ExplanationOfBenefitStatus>(this, ExplanationOfBenefitStatus.CANCELLED, code);
      if ("draft".equals(codeString))
        return new Enumeration<ExplanationOfBenefitStatus>(this, ExplanationOfBenefitStatus.DRAFT, code);
      if ("entered-in-error".equals(codeString))
        return new Enumeration<ExplanationOfBenefitStatus>(this, ExplanationOfBenefitStatus.ENTEREDINERROR, code);
      throw new FHIRException("Unknown ExplanationOfBenefitStatus code '" + codeString + "'");
    }

    public String toCode(ExplanationOfBenefitStatus code) {
       if (code == ExplanationOfBenefitStatus.NULL)
           return null;
       if (code == ExplanationOfBenefitStatus.ACTIVE)
        return "active";
      if (code == ExplanationOfBenefitStatus.CANCELLED)
        return "cancelled";
      if (code == ExplanationOfBenefitStatus.DRAFT)
        return "draft";
      if (code == ExplanationOfBenefitStatus.ENTEREDINERROR)
        return "entered-in-error";
      return "?";
   }

    public String toSystem(ExplanationOfBenefitStatus code) {
      return code.getSystem();
    }
  }

  @Block()
  public static class RelatedClaimComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Reference to a related claim.
     */
    @Child(name = "claim", type = { Claim.class }, order = 1, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Reference to the related claim", formalDefinition = "Reference to a related claim.")
    protected Reference claim;

    /**
     * A code to convey how the claims are related.
     */
    @Child(name = "relationship", type = {
        CodeableConcept.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "How the reference claim is related", formalDefinition = "A code to convey how the claims are related.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/related-claim-relationship")
    protected CodeableConcept relationship;

    /**
     * An alternate organizational reference to the case or file to which this
     * particular claim pertains.
     */
    @Child(name = "reference", type = {
        Identifier.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "File or case reference", formalDefinition = "An alternate organizational reference to the case or file to which this particular claim pertains.")
    protected Identifier reference;

    private static final long serialVersionUID = 1047077926L;

    /**
     * Constructor
     */
    public RelatedClaimComponent() {
      super();
    }

    /**
     * @return {@link #claim} (Reference to a related claim.)
     */
    public Reference getClaim() {
      if (this.claim == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create RelatedClaimComponent.claim");
        else if (Configuration.doAutoCreate())
          this.claim = new Reference(); // cc
      return this.claim;
    }

    public boolean hasClaim() {
      return this.claim != null && !this.claim.isEmpty();
    }

    /**
     * @param value {@link #claim} (Reference to a related claim.)
     */
    public RelatedClaimComponent setClaim(Reference value) {
      this.claim = value;
      return this;
    }

    /**
     * @return {@link #relationship} (A code to convey how the claims are related.)
     */
    public CodeableConcept getRelationship() {
      if (this.relationship == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create RelatedClaimComponent.relationship");
        else if (Configuration.doAutoCreate())
          this.relationship = new CodeableConcept(); // cc
      return this.relationship;
    }

    public boolean hasRelationship() {
      return this.relationship != null && !this.relationship.isEmpty();
    }

    /**
     * @param value {@link #relationship} (A code to convey how the claims are
     *              related.)
     */
    public RelatedClaimComponent setRelationship(CodeableConcept value) {
      this.relationship = value;
      return this;
    }

    /**
     * @return {@link #reference} (An alternate organizational reference to the case
     *         or file to which this particular claim pertains.)
     */
    public Identifier getReference() {
      if (this.reference == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create RelatedClaimComponent.reference");
        else if (Configuration.doAutoCreate())
          this.reference = new Identifier(); // cc
      return this.reference;
    }

    public boolean hasReference() {
      return this.reference != null && !this.reference.isEmpty();
    }

    /**
     * @param value {@link #reference} (An alternate organizational reference to the
     *              case or file to which this particular claim pertains.)
     */
    public RelatedClaimComponent setReference(Identifier value) {
      this.reference = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("claim", "Reference(Claim)", "Reference to a related claim.", 0, 1, claim));
      children.add(new Property("relationship", "CodeableConcept", "A code to convey how the claims are related.", 0, 1,
          relationship));
      children.add(new Property("reference", "Identifier",
          "An alternate organizational reference to the case or file to which this particular claim pertains.", 0, 1,
          reference));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 94742588:
        /* claim */ return new Property("claim", "Reference(Claim)", "Reference to a related claim.", 0, 1, claim);
      case -261851592:
        /* relationship */ return new Property("relationship", "CodeableConcept",
            "A code to convey how the claims are related.", 0, 1, relationship);
      case -925155509:
        /* reference */ return new Property("reference", "Identifier",
            "An alternate organizational reference to the case or file to which this particular claim pertains.", 0, 1,
            reference);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 94742588:
        /* claim */ return this.claim == null ? new Base[0] : new Base[] { this.claim }; // Reference
      case -261851592:
        /* relationship */ return this.relationship == null ? new Base[0] : new Base[] { this.relationship }; // CodeableConcept
      case -925155509:
        /* reference */ return this.reference == null ? new Base[0] : new Base[] { this.reference }; // Identifier
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 94742588: // claim
        this.claim = TypeConvertor.castToReference(value); // Reference
        return value;
      case -261851592: // relationship
        this.relationship = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -925155509: // reference
        this.reference = TypeConvertor.castToIdentifier(value); // Identifier
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("claim")) {
        this.claim = TypeConvertor.castToReference(value); // Reference
      } else if (name.equals("relationship")) {
        this.relationship = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("reference")) {
        this.reference = TypeConvertor.castToIdentifier(value); // Identifier
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("claim")) {
        this.claim = null;
      } else if (name.equals("relationship")) {
        this.relationship = null;
      } else if (name.equals("reference")) {
        this.reference = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 94742588:
        return getClaim();
      case -261851592:
        return getRelationship();
      case -925155509:
        return getReference();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 94742588:
        /* claim */ return new String[] { "Reference" };
      case -261851592:
        /* relationship */ return new String[] { "CodeableConcept" };
      case -925155509:
        /* reference */ return new String[] { "Identifier" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("claim")) {
        this.claim = new Reference();
        return this.claim;
      } else if (name.equals("relationship")) {
        this.relationship = new CodeableConcept();
        return this.relationship;
      } else if (name.equals("reference")) {
        this.reference = new Identifier();
        return this.reference;
      } else
        return super.addChild(name);
    }

    public RelatedClaimComponent copy() {
      RelatedClaimComponent dst = new RelatedClaimComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(RelatedClaimComponent dst) {
      super.copyValues(dst);
      dst.claim = claim == null ? null : claim.copy();
      dst.relationship = relationship == null ? null : relationship.copy();
      dst.reference = reference == null ? null : reference.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof RelatedClaimComponent))
        return false;
      RelatedClaimComponent o = (RelatedClaimComponent) other_;
      return compareDeep(claim, o.claim, true) && compareDeep(relationship, o.relationship, true)
          && compareDeep(reference, o.reference, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof RelatedClaimComponent))
        return false;
      RelatedClaimComponent o = (RelatedClaimComponent) other_;
      return true;
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(claim, relationship, reference);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.related";

    }

  }

  @Block()
  public static class PayeeComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Type of Party to be reimbursed: Subscriber, provider, other.
     */
    @Child(name = "type", type = {
        CodeableConcept.class }, order = 1, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Category of recipient", formalDefinition = "Type of Party to be reimbursed: Subscriber, provider, other.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/payeetype")
    protected CodeableConcept type;

    /**
     * Reference to the individual or organization to whom any payment will be made.
     */
    @Child(name = "party", type = { Practitioner.class, PractitionerRole.class, Organization.class, Patient.class,
        RelatedPerson.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Recipient reference", formalDefinition = "Reference to the individual or organization to whom any payment will be made.")
    protected Reference party;

    private static final long serialVersionUID = -1948897146L;

    /**
     * Constructor
     */
    public PayeeComponent() {
      super();
    }

    /**
     * @return {@link #type} (Type of Party to be reimbursed: Subscriber, provider,
     *         other.)
     */
    public CodeableConcept getType() {
      if (this.type == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PayeeComponent.type");
        else if (Configuration.doAutoCreate())
          this.type = new CodeableConcept(); // cc
      return this.type;
    }

    public boolean hasType() {
      return this.type != null && !this.type.isEmpty();
    }

    /**
     * @param value {@link #type} (Type of Party to be reimbursed: Subscriber,
     *              provider, other.)
     */
    public PayeeComponent setType(CodeableConcept value) {
      this.type = value;
      return this;
    }

    /**
     * @return {@link #party} (Reference to the individual or organization to whom
     *         any payment will be made.)
     */
    public Reference getParty() {
      if (this.party == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PayeeComponent.party");
        else if (Configuration.doAutoCreate())
          this.party = new Reference(); // cc
      return this.party;
    }

    public boolean hasParty() {
      return this.party != null && !this.party.isEmpty();
    }

    /**
     * @param value {@link #party} (Reference to the individual or organization to
     *              whom any payment will be made.)
     */
    public PayeeComponent setParty(Reference value) {
      this.party = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("type", "CodeableConcept",
          "Type of Party to be reimbursed: Subscriber, provider, other.", 0, 1, type));
      children.add(new Property("party", "Reference(Practitioner|PractitionerRole|Organization|Patient|RelatedPerson)",
          "Reference to the individual or organization to whom any payment will be made.", 0, 1, party));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 3575610:
        /* type */ return new Property("type", "CodeableConcept",
            "Type of Party to be reimbursed: Subscriber, provider, other.", 0, 1, type);
      case 106437350:
        /* party */ return new Property("party",
            "Reference(Practitioner|PractitionerRole|Organization|Patient|RelatedPerson)",
            "Reference to the individual or organization to whom any payment will be made.", 0, 1, party);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : new Base[] { this.type }; // CodeableConcept
      case 106437350:
        /* party */ return this.party == null ? new Base[0] : new Base[] { this.party }; // Reference
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 3575610: // type
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 106437350: // party
        this.party = TypeConvertor.castToReference(value); // Reference
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("type")) {
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("party")) {
        this.party = TypeConvertor.castToReference(value); // Reference
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("type")) {
        this.type = null;
      } else if (name.equals("party")) {
        this.party = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3575610:
        return getType();
      case 106437350:
        return getParty();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3575610:
        /* type */ return new String[] { "CodeableConcept" };
      case 106437350:
        /* party */ return new String[] { "Reference" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("type")) {
        this.type = new CodeableConcept();
        return this.type;
      } else if (name.equals("party")) {
        this.party = new Reference();
        return this.party;
      } else
        return super.addChild(name);
    }

    public PayeeComponent copy() {
      PayeeComponent dst = new PayeeComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(PayeeComponent dst) {
      super.copyValues(dst);
      dst.type = type == null ? null : type.copy();
      dst.party = party == null ? null : party.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof PayeeComponent))
        return false;
      PayeeComponent o = (PayeeComponent) other_;
      return compareDeep(type, o.type, true) && compareDeep(party, o.party, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof PayeeComponent))
        return false;
      PayeeComponent o = (PayeeComponent) other_;
      return true;
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(type, party);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.payee";

    }

  }

  @Block()
  public static class CareTeamComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A number to uniquely identify care team entries.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Order of care team", formalDefinition = "A number to uniquely identify care team entries.")
    protected PositiveIntType sequence;

    /**
     * Member of the team who provided the product or service.
     */
    @Child(name = "provider", type = { Practitioner.class, PractitionerRole.class,
        Organization.class }, order = 2, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Practitioner or organization", formalDefinition = "Member of the team who provided the product or service.")
    protected Reference provider;

    /**
     * The party who is billing and/or responsible for the claimed products or
     * services.
     */
    @Child(name = "responsible", type = {
        BooleanType.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Indicator of the lead practitioner", formalDefinition = "The party who is billing and/or responsible for the claimed products or services.")
    protected BooleanType responsible;

    /**
     * The lead, assisting or supervising practitioner and their discipline if a
     * multidisciplinary team.
     */
    @Child(name = "role", type = {
        CodeableConcept.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Function within the team", formalDefinition = "The lead, assisting or supervising practitioner and their discipline if a multidisciplinary team.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-careteamrole")
    protected CodeableConcept role;

    /**
     * The qualification of the practitioner which is applicable for this service.
     */
    @Child(name = "qualification", type = {
        CodeableConcept.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Practitioner credential or specialization", formalDefinition = "The qualification of the practitioner which is applicable for this service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/provider-qualification")
    protected CodeableConcept qualification;

    private static final long serialVersionUID = 1479624238L;

    /**
     * Constructor
     */
    public CareTeamComponent() {
      super();
    }

    /**
     * Constructor
     */
    public CareTeamComponent(int sequence, Reference provider) {
      super();
      this.setSequence(sequence);
      this.setProvider(provider);
    }

    /**
     * @return {@link #sequence} (A number to uniquely identify care team entries.).
     *         This is the underlying object with id, value and extensions. The
     *         accessor "getSequence" gives direct access to the value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create CareTeamComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A number to uniquely identify care team
     *              entries.). This is the underlying object with id, value and
     *              extensions. The accessor "getSequence" gives direct access to
     *              the value
     */
    public CareTeamComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A number to uniquely identify care team entries.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A number to uniquely identify care team entries.
     */
    public CareTeamComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #provider} (Member of the team who provided the product or
     *         service.)
     */
    public Reference getProvider() {
      if (this.provider == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create CareTeamComponent.provider");
        else if (Configuration.doAutoCreate())
          this.provider = new Reference(); // cc
      return this.provider;
    }

    public boolean hasProvider() {
      return this.provider != null && !this.provider.isEmpty();
    }

    /**
     * @param value {@link #provider} (Member of the team who provided the product
     *              or service.)
     */
    public CareTeamComponent setProvider(Reference value) {
      this.provider = value;
      return this;
    }

    /**
     * @return {@link #responsible} (The party who is billing and/or responsible for
     *         the claimed products or services.). This is the underlying object
     *         with id, value and extensions. The accessor "getResponsible" gives
     *         direct access to the value
     */
    public BooleanType getResponsibleElement() {
      if (this.responsible == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create CareTeamComponent.responsible");
        else if (Configuration.doAutoCreate())
          this.responsible = new BooleanType(); // bb
      return this.responsible;
    }

    public boolean hasResponsibleElement() {
      return this.responsible != null && !this.responsible.isEmpty();
    }

    public boolean hasResponsible() {
      return this.responsible != null && !this.responsible.isEmpty();
    }

    /**
     * @param value {@link #responsible} (The party who is billing and/or
     *              responsible for the claimed products or services.). This is the
     *              underlying object with id, value and extensions. The accessor
     *              "getResponsible" gives direct access to the value
     */
    public CareTeamComponent setResponsibleElement(BooleanType value) {
      this.responsible = value;
      return this;
    }

    /**
     * @return The party who is billing and/or responsible for the claimed products
     *         or services.
     */
    public boolean getResponsible() {
      return this.responsible == null || this.responsible.isEmpty() ? false : this.responsible.getValue();
    }

    /**
     * @param value The party who is billing and/or responsible for the claimed
     *              products or services.
     */
    public CareTeamComponent setResponsible(boolean value) {
      if (this.responsible == null)
        this.responsible = new BooleanType();
      this.responsible.setValue(value);
      return this;
    }

    /**
     * @return {@link #role} (The lead, assisting or supervising practitioner and
     *         their discipline if a multidisciplinary team.)
     */
    public CodeableConcept getRole() {
      if (this.role == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create CareTeamComponent.role");
        else if (Configuration.doAutoCreate())
          this.role = new CodeableConcept(); // cc
      return this.role;
    }

    public boolean hasRole() {
      return this.role != null && !this.role.isEmpty();
    }

    /**
     * @param value {@link #role} (The lead, assisting or supervising practitioner
     *              and their discipline if a multidisciplinary team.)
     */
    public CareTeamComponent setRole(CodeableConcept value) {
      this.role = value;
      return this;
    }

    /**
     * @return {@link #qualification} (The qualification of the practitioner which
     *         is applicable for this service.)
     */
    public CodeableConcept getQualification() {
      if (this.qualification == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create CareTeamComponent.qualification");
        else if (Configuration.doAutoCreate())
          this.qualification = new CodeableConcept(); // cc
      return this.qualification;
    }

    public boolean hasQualification() {
      return this.qualification != null && !this.qualification.isEmpty();
    }

    /**
     * @param value {@link #qualification} (The qualification of the practitioner
     *              which is applicable for this service.)
     */
    public CareTeamComponent setQualification(CodeableConcept value) {
      this.qualification = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(
          new Property("sequence", "positiveInt", "A number to uniquely identify care team entries.", 0, 1, sequence));
      children.add(new Property("provider", "Reference(Practitioner|PractitionerRole|Organization)",
          "Member of the team who provided the product or service.", 0, 1, provider));
      children.add(new Property("responsible", "boolean",
          "The party who is billing and/or responsible for the claimed products or services.", 0, 1, responsible));
      children.add(new Property("role", "CodeableConcept",
          "The lead, assisting or supervising practitioner and their discipline if a multidisciplinary team.", 0, 1,
          role));
      children.add(new Property("qualification", "CodeableConcept",
          "The qualification of the practitioner which is applicable for this service.", 0, 1, qualification));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt",
            "A number to uniquely identify care team entries.", 0, 1, sequence);
      case -987494927:
        /* provider */ return new Property("provider", "Reference(Practitioner|PractitionerRole|Organization)",
            "Member of the team who provided the product or service.", 0, 1, provider);
      case 1847674614:
        /* responsible */ return new Property("responsible", "boolean",
            "The party who is billing and/or responsible for the claimed products or services.", 0, 1, responsible);
      case 3506294:
        /* role */ return new Property("role", "CodeableConcept",
            "The lead, assisting or supervising practitioner and their discipline if a multidisciplinary team.", 0, 1,
            role);
      case -631333393:
        /* qualification */ return new Property("qualification", "CodeableConcept",
            "The qualification of the practitioner which is applicable for this service.", 0, 1, qualification);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case -987494927:
        /* provider */ return this.provider == null ? new Base[0] : new Base[] { this.provider }; // Reference
      case 1847674614:
        /* responsible */ return this.responsible == null ? new Base[0] : new Base[] { this.responsible }; // BooleanType
      case 3506294:
        /* role */ return this.role == null ? new Base[0] : new Base[] { this.role }; // CodeableConcept
      case -631333393:
        /* qualification */ return this.qualification == null ? new Base[0] : new Base[] { this.qualification }; // CodeableConcept
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case -987494927: // provider
        this.provider = TypeConvertor.castToReference(value); // Reference
        return value;
      case 1847674614: // responsible
        this.responsible = TypeConvertor.castToBoolean(value); // BooleanType
        return value;
      case 3506294: // role
        this.role = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -631333393: // qualification
        this.qualification = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("provider")) {
        this.provider = TypeConvertor.castToReference(value); // Reference
      } else if (name.equals("responsible")) {
        this.responsible = TypeConvertor.castToBoolean(value); // BooleanType
      } else if (name.equals("role")) {
        this.role = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("qualification")) {
        this.qualification = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("provider")) {
        this.provider = null;
      } else if (name.equals("responsible")) {
        this.responsible = null;
      } else if (name.equals("role")) {
        this.role = null;
      } else if (name.equals("qualification")) {
        this.qualification = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case -987494927:
        return getProvider();
      case 1847674614:
        return getResponsibleElement();
      case 3506294:
        return getRole();
      case -631333393:
        return getQualification();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case -987494927:
        /* provider */ return new String[] { "Reference" };
      case 1847674614:
        /* responsible */ return new String[] { "boolean" };
      case 3506294:
        /* role */ return new String[] { "CodeableConcept" };
      case -631333393:
        /* qualification */ return new String[] { "CodeableConcept" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.careTeam.sequence");
      } else if (name.equals("provider")) {
        this.provider = new Reference();
        return this.provider;
      } else if (name.equals("responsible")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.careTeam.responsible");
      } else if (name.equals("role")) {
        this.role = new CodeableConcept();
        return this.role;
      } else if (name.equals("qualification")) {
        this.qualification = new CodeableConcept();
        return this.qualification;
      } else
        return super.addChild(name);
    }

    public CareTeamComponent copy() {
      CareTeamComponent dst = new CareTeamComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(CareTeamComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      dst.provider = provider == null ? null : provider.copy();
      dst.responsible = responsible == null ? null : responsible.copy();
      dst.role = role == null ? null : role.copy();
      dst.qualification = qualification == null ? null : qualification.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof CareTeamComponent))
        return false;
      CareTeamComponent o = (CareTeamComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(provider, o.provider, true)
          && compareDeep(responsible, o.responsible, true) && compareDeep(role, o.role, true)
          && compareDeep(qualification, o.qualification, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof CareTeamComponent))
        return false;
      CareTeamComponent o = (CareTeamComponent) other_;
      return compareValues(sequence, o.sequence, true) && compareValues(responsible, o.responsible, true);
    }

    public boolean isEmpty() {
      return super.isEmpty()
          && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, provider, responsible, role, qualification);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.careTeam";

    }

  }

  @Block()
  public static class SupportingInformationComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A number to uniquely identify supporting information entries.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Information instance identifier", formalDefinition = "A number to uniquely identify supporting information entries.")
    protected PositiveIntType sequence;

    /**
     * The general class of the information supplied: information; exception;
     * accident, employment; onset, etc.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 2, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Classification of the supplied information", formalDefinition = "The general class of the information supplied: information; exception; accident, employment; onset, etc.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-informationcategory")
    protected CodeableConcept category;

    /**
     * System and code pertaining to the specific information regarding special
     * conditions relating to the setting, treatment or patient for which care is
     * sought.
     */
    @Child(name = "code", type = {
        CodeableConcept.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Type of information", formalDefinition = "System and code pertaining to the specific information regarding special conditions relating to the setting, treatment or patient  for which care is sought.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-exception")
    protected CodeableConcept code;

    /**
     * The date when or period to which this information refers.
     */
    @Child(name = "timing", type = { DateType.class,
        Period.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "When it occurred", formalDefinition = "The date when or period to which this information refers.")
    protected DataType timing;

    /**
     * Additional data or information such as resources, documents, images etc.
     * including references to the data or the actual inclusion of the data.
     */
    @Child(name = "value", type = { BooleanType.class, StringType.class, Quantity.class, Attachment.class,
        Reference.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Data to be provided", formalDefinition = "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.")
    protected DataType value;

    /**
     * Provides the reason in the situation where a reason code is required in
     * addition to the content.
     */
    @Child(name = "reason", type = { Coding.class }, order = 6, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Explanation for the information", formalDefinition = "Provides the reason in the situation where a reason code is required in addition to the content.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/missing-tooth-reason")
    protected Coding reason;

    private static final long serialVersionUID = 1577205655L;

    /**
     * Constructor
     */
    public SupportingInformationComponent() {
      super();
    }

    /**
     * Constructor
     */
    public SupportingInformationComponent(int sequence, CodeableConcept category) {
      super();
      this.setSequence(sequence);
      this.setCategory(category);
    }

    /**
     * @return {@link #sequence} (A number to uniquely identify supporting
     *         information entries.). This is the underlying object with id, value
     *         and extensions. The accessor "getSequence" gives direct access to the
     *         value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SupportingInformationComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A number to uniquely identify supporting
     *              information entries.). This is the underlying object with id,
     *              value and extensions. The accessor "getSequence" gives direct
     *              access to the value
     */
    public SupportingInformationComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A number to uniquely identify supporting information entries.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A number to uniquely identify supporting information entries.
     */
    public SupportingInformationComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #category} (The general class of the information supplied:
     *         information; exception; accident, employment; onset, etc.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SupportingInformationComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (The general class of the information
     *              supplied: information; exception; accident, employment; onset,
     *              etc.)
     */
    public SupportingInformationComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #code} (System and code pertaining to the specific information
     *         regarding special conditions relating to the setting, treatment or
     *         patient for which care is sought.)
     */
    public CodeableConcept getCode() {
      if (this.code == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SupportingInformationComponent.code");
        else if (Configuration.doAutoCreate())
          this.code = new CodeableConcept(); // cc
      return this.code;
    }

    public boolean hasCode() {
      return this.code != null && !this.code.isEmpty();
    }

    /**
     * @param value {@link #code} (System and code pertaining to the specific
     *              information regarding special conditions relating to the
     *              setting, treatment or patient for which care is sought.)
     */
    public SupportingInformationComponent setCode(CodeableConcept value) {
      this.code = value;
      return this;
    }

    /**
     * @return {@link #timing} (The date when or period to which this information
     *         refers.)
     */
    public DataType getTiming() {
      return this.timing;
    }

    /**
     * @return {@link #timing} (The date when or period to which this information
     *         refers.)
     */
    public DateType getTimingDateType() throws FHIRException {
      if (this.timing == null)
        this.timing = new DateType();
      if (!(this.timing instanceof DateType))
        throw new FHIRException("Type mismatch: the type DateType was expected, but " + this.timing.getClass().getName()
            + " was encountered");
      return (DateType) this.timing;
    }

    public boolean hasTimingDateType() {
        return this.timing instanceof DateType;
    }

    /**
     * @return {@link #timing} (The date when or period to which this information
     *         refers.)
     */
    public Period getTimingPeriod() throws FHIRException {
      if (this.timing == null)
        this.timing = new Period();
      if (!(this.timing instanceof Period))
        throw new FHIRException("Type mismatch: the type Period was expected, but " + this.timing.getClass().getName()
            + " was encountered");
      return (Period) this.timing;
    }

    public boolean hasTimingPeriod() {
        return this.timing instanceof Period;
    }

    public boolean hasTiming() {
      return this.timing != null && !this.timing.isEmpty();
    }

    /**
     * @param value {@link #timing} (The date when or period to which this
     *              information refers.)
     */
    public SupportingInformationComponent setTiming(DataType value) {
      if (value != null && !(value instanceof DateType || value instanceof Period))
        throw new Error("Not the right type for ExplanationOfBenefit.supportingInfo.timing[x]: " + value.fhirType());
      this.timing = value;
      return this;
    }

    /**
     * @return {@link #value} (Additional data or information such as resources,
     *         documents, images etc. including references to the data or the actual
     *         inclusion of the data.)
     */
    public DataType getValue() {
      return this.value;
    }

    /**
     * @return {@link #value} (Additional data or information such as resources,
     *         documents, images etc. including references to the data or the actual
     *         inclusion of the data.)
     */
    public BooleanType getValueBooleanType() throws FHIRException {
      if (this.value == null)
        this.value = new BooleanType();
      if (!(this.value instanceof BooleanType))
        throw new FHIRException("Type mismatch: the type BooleanType was expected, but "
            + this.value.getClass().getName() + " was encountered");
      return (BooleanType) this.value;
    }

    public boolean hasValueBooleanType() {
        return this.value instanceof BooleanType;
    }

    /**
     * @return {@link #value} (Additional data or information such as resources,
     *         documents, images etc. including references to the data or the actual
     *         inclusion of the data.)
     */
    public StringType getValueStringType() throws FHIRException {
      if (this.value == null)
        this.value = new StringType();
      if (!(this.value instanceof StringType))
        throw new FHIRException("Type mismatch: the type StringType was expected, but "
            + this.value.getClass().getName() + " was encountered");
      return (StringType) this.value;
    }

    public boolean hasValueStringType() {
        return this.value instanceof StringType;
    }

    /**
     * @return {@link #value} (Additional data or information such as resources,
     *         documents, images etc. including references to the data or the actual
     *         inclusion of the data.)
     */
    public Quantity getValueQuantity() throws FHIRException {
      if (this.value == null)
        this.value = new Quantity();
      if (!(this.value instanceof Quantity))
        throw new FHIRException("Type mismatch: the type Quantity was expected, but " + this.value.getClass().getName()
            + " was encountered");
      return (Quantity) this.value;
    }

    public boolean hasValueQuantity() {
        return this.value instanceof Quantity;
    }

    /**
     * @return {@link #value} (Additional data or information such as resources,
     *         documents, images etc. including references to the data or the actual
     *         inclusion of the data.)
     */
    public Attachment getValueAttachment() throws FHIRException {
      if (this.value == null)
        this.value = new Attachment();
      if (!(this.value instanceof Attachment))
        throw new FHIRException("Type mismatch: the type Attachment was expected, but "
            + this.value.getClass().getName() + " was encountered");
      return (Attachment) this.value;
    }

    public boolean hasValueAttachment() {
        return this.value instanceof Attachment;
    }

    /**
     * @return {@link #value} (Additional data or information such as resources,
     *         documents, images etc. including references to the data or the actual
     *         inclusion of the data.)
     */
    public Reference getValueReference() throws FHIRException {
      if (this.value == null)
        this.value = new Reference();
      if (!(this.value instanceof Reference))
        throw new FHIRException("Type mismatch: the type Reference was expected, but " + this.value.getClass().getName()
            + " was encountered");
      return (Reference) this.value;
    }

    public boolean hasValueReference() {
        return this.value instanceof Reference;
    }

    public boolean hasValue() {
      return this.value != null && !this.value.isEmpty();
    }

    /**
     * @param value {@link #value} (Additional data or information such as
     *              resources, documents, images etc. including references to the
     *              data or the actual inclusion of the data.)
     */
    public SupportingInformationComponent setValue(DataType value) {
      if (value != null && !(value instanceof BooleanType || value instanceof StringType || value instanceof Quantity
          || value instanceof Attachment || value instanceof Reference))
        throw new Error("Not the right type for ExplanationOfBenefit.supportingInfo.value[x]: " + value.fhirType());
      this.value = value;
      return this;
    }

    /**
     * @return {@link #reason} (Provides the reason in the situation where a reason
     *         code is required in addition to the content.)
     */
    public Coding getReason() {
      if (this.reason == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SupportingInformationComponent.reason");
        else if (Configuration.doAutoCreate())
          this.reason = new Coding(); // cc
      return this.reason;
    }

    public boolean hasReason() {
      return this.reason != null && !this.reason.isEmpty();
    }

    /**
     * @param value {@link #reason} (Provides the reason in the situation where a
     *              reason code is required in addition to the content.)
     */
    public SupportingInformationComponent setReason(Coding value) {
      this.reason = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("sequence", "positiveInt",
          "A number to uniquely identify supporting information entries.", 0, 1, sequence));
      children.add(new Property("category", "CodeableConcept",
          "The general class of the information supplied: information; exception; accident, employment; onset, etc.", 0,
          1, category));
      children.add(new Property("code", "CodeableConcept",
          "System and code pertaining to the specific information regarding special conditions relating to the setting, treatment or patient  for which care is sought.",
          0, 1, code));
      children.add(new Property("timing[x]", "date|Period", "The date when or period to which this information refers.",
          0, 1, timing));
      children.add(new Property("value[x]", "boolean|string|Quantity|Attachment|Reference(Any)",
          "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
          0, 1, value));
      children.add(new Property("reason", "Coding",
          "Provides the reason in the situation where a reason code is required in addition to the content.", 0, 1,
          reason));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt",
            "A number to uniquely identify supporting information entries.", 0, 1, sequence);
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "The general class of the information supplied: information; exception; accident, employment; onset, etc.",
            0, 1, category);
      case 3059181:
        /* code */ return new Property("code", "CodeableConcept",
            "System and code pertaining to the specific information regarding special conditions relating to the setting, treatment or patient  for which care is sought.",
            0, 1, code);
      case 164632566:
        /* timing[x] */ return new Property("timing[x]", "date|Period",
            "The date when or period to which this information refers.", 0, 1, timing);
      case -873664438:
        /* timing */ return new Property("timing[x]", "date|Period",
            "The date when or period to which this information refers.", 0, 1, timing);
      case 807935768:
        /* timingDate */ return new Property("timing[x]", "date",
            "The date when or period to which this information refers.", 0, 1, timing);
      case -615615829:
        /* timingPeriod */ return new Property("timing[x]", "Period",
            "The date when or period to which this information refers.", 0, 1, timing);
      case -1410166417:
        /* value[x] */ return new Property("value[x]", "boolean|string|Quantity|Attachment|Reference(Any)",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case 111972721:
        /* value */ return new Property("value[x]", "boolean|string|Quantity|Attachment|Reference(Any)",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case 733421943:
        /* valueBoolean */ return new Property("value[x]", "boolean",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case -1424603934:
        /* valueString */ return new Property("value[x]", "string",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case -2029823716:
        /* valueQuantity */ return new Property("value[x]", "Quantity",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case -475566732:
        /* valueAttachment */ return new Property("value[x]", "Attachment",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case 1755241690:
        /* valueReference */ return new Property("value[x]", "Reference(Any)",
            "Additional data or information such as resources, documents, images etc. including references to the data or the actual inclusion of the data.",
            0, 1, value);
      case -934964668:
        /* reason */ return new Property("reason", "Coding",
            "Provides the reason in the situation where a reason code is required in addition to the content.", 0, 1,
            reason);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case 3059181:
        /* code */ return this.code == null ? new Base[0] : new Base[] { this.code }; // CodeableConcept
      case -873664438:
        /* timing */ return this.timing == null ? new Base[0] : new Base[] { this.timing }; // DataType
      case 111972721:
        /* value */ return this.value == null ? new Base[0] : new Base[] { this.value }; // DataType
      case -934964668:
        /* reason */ return this.reason == null ? new Base[0] : new Base[] { this.reason }; // Coding
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 3059181: // code
        this.code = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -873664438: // timing
        this.timing = TypeConvertor.castToType(value); // DataType
        return value;
      case 111972721: // value
        this.value = TypeConvertor.castToType(value); // DataType
        return value;
      case -934964668: // reason
        this.reason = TypeConvertor.castToCoding(value); // Coding
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("code")) {
        this.code = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("timing[x]")) {
        this.timing = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("value[x]")) {
        this.value = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("reason")) {
        this.reason = TypeConvertor.castToCoding(value); // Coding
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("code")) {
        this.code = null;
      } else if (name.equals("timing[x]")) {
        this.timing = null;
      } else if (name.equals("value[x]")) {
        this.value = null;
      } else if (name.equals("reason")) {
        this.reason = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case 50511102:
        return getCategory();
      case 3059181:
        return getCode();
      case 164632566:
        return getTiming();
      case -873664438:
        return getTiming();
      case -1410166417:
        return getValue();
      case 111972721:
        return getValue();
      case -934964668:
        return getReason();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case 3059181:
        /* code */ return new String[] { "CodeableConcept" };
      case -873664438:
        /* timing */ return new String[] { "date", "Period" };
      case 111972721:
        /* value */ return new String[] { "boolean", "string", "Quantity", "Attachment", "Reference" };
      case -934964668:
        /* reason */ return new String[] { "Coding" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.supportingInfo.sequence");
      } else if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("code")) {
        this.code = new CodeableConcept();
        return this.code;
      } else if (name.equals("timingDate")) {
        this.timing = new DateType();
        return this.timing;
      } else if (name.equals("timingPeriod")) {
        this.timing = new Period();
        return this.timing;
      } else if (name.equals("valueBoolean")) {
        this.value = new BooleanType();
        return this.value;
      } else if (name.equals("valueString")) {
        this.value = new StringType();
        return this.value;
      } else if (name.equals("valueQuantity")) {
        this.value = new Quantity();
        return this.value;
      } else if (name.equals("valueAttachment")) {
        this.value = new Attachment();
        return this.value;
      } else if (name.equals("valueReference")) {
        this.value = new Reference();
        return this.value;
      } else if (name.equals("reason")) {
        this.reason = new Coding();
        return this.reason;
      } else
        return super.addChild(name);
    }

    public SupportingInformationComponent copy() {
      SupportingInformationComponent dst = new SupportingInformationComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(SupportingInformationComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      dst.category = category == null ? null : category.copy();
      dst.code = code == null ? null : code.copy();
      dst.timing = timing == null ? null : timing.copy();
      dst.value = value == null ? null : value.copy();
      dst.reason = reason == null ? null : reason.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof SupportingInformationComponent))
        return false;
      SupportingInformationComponent o = (SupportingInformationComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(category, o.category, true)
          && compareDeep(code, o.code, true) && compareDeep(timing, o.timing, true) && compareDeep(value, o.value, true)
          && compareDeep(reason, o.reason, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof SupportingInformationComponent))
        return false;
      SupportingInformationComponent o = (SupportingInformationComponent) other_;
      return compareValues(sequence, o.sequence, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, category, code, timing, value, reason);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.supportingInfo";

    }

  }

  @Block()
  public static class DiagnosisComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A number to uniquely identify diagnosis entries.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Diagnosis instance identifier", formalDefinition = "A number to uniquely identify diagnosis entries.")
    protected PositiveIntType sequence;

    /**
     * The nature of illness or problem in a coded form or as a reference to an
     * external defined Condition.
     */
    @Child(name = "diagnosis", type = { CodeableConcept.class,
        Condition.class }, order = 2, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Nature of illness or problem", formalDefinition = "The nature of illness or problem in a coded form or as a reference to an external defined Condition.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/icd-10")
    protected DataType diagnosis;

    /**
     * When the condition was observed or the relative ranking.
     */
    @Child(name = "type", type = {
        CodeableConcept.class }, order = 3, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Timing or nature of the diagnosis", formalDefinition = "When the condition was observed or the relative ranking.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-diagnosistype")
    protected List<CodeableConcept> type;

    /**
     * Indication of whether the diagnosis was present on admission to a facility.
     */
    @Child(name = "onAdmission", type = {
        CodeableConcept.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Present on admission", formalDefinition = "Indication of whether the diagnosis was present on admission to a facility.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-diagnosis-on-admission")
    protected CodeableConcept onAdmission;

    /**
     * A package billing code or bundle code used to group products and services to
     * a particular health condition (such as heart attack) which is based on a
     * predetermined grouping code system.
     */
    @Child(name = "packageCode", type = {
        CodeableConcept.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Package billing code", formalDefinition = "A package billing code or bundle code used to group products and services to a particular health condition (such as heart attack) which is based on a predetermined grouping code system.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-diagnosisrelatedgroup")
    protected CodeableConcept packageCode;

    private static final long serialVersionUID = 550515328L;

    /**
     * Constructor
     */
    public DiagnosisComponent() {
      super();
    }

    /**
     * Constructor
     */
    public DiagnosisComponent(int sequence, DataType diagnosis) {
      super();
      this.setSequence(sequence);
      this.setDiagnosis(diagnosis);
    }

    /**
     * @return {@link #sequence} (A number to uniquely identify diagnosis entries.).
     *         This is the underlying object with id, value and extensions. The
     *         accessor "getSequence" gives direct access to the value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DiagnosisComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A number to uniquely identify diagnosis
     *              entries.). This is the underlying object with id, value and
     *              extensions. The accessor "getSequence" gives direct access to
     *              the value
     */
    public DiagnosisComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A number to uniquely identify diagnosis entries.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A number to uniquely identify diagnosis entries.
     */
    public DiagnosisComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #diagnosis} (The nature of illness or problem in a coded form
     *         or as a reference to an external defined Condition.)
     */
    public DataType getDiagnosis() {
      return this.diagnosis;
    }

    /**
     * @return {@link #diagnosis} (The nature of illness or problem in a coded form
     *         or as a reference to an external defined Condition.)
     */
    public CodeableConcept getDiagnosisCodeableConcept() throws FHIRException {
      if (this.diagnosis == null)
        this.diagnosis = new CodeableConcept();
      if (!(this.diagnosis instanceof CodeableConcept))
        throw new FHIRException("Type mismatch: the type CodeableConcept was expected, but "
            + this.diagnosis.getClass().getName() + " was encountered");
      return (CodeableConcept) this.diagnosis;
    }

    public boolean hasDiagnosisCodeableConcept() {
        return this.diagnosis instanceof CodeableConcept;
    }

    /**
     * @return {@link #diagnosis} (The nature of illness or problem in a coded form
     *         or as a reference to an external defined Condition.)
     */
    public Reference getDiagnosisReference() throws FHIRException {
      if (this.diagnosis == null)
        this.diagnosis = new Reference();
      if (!(this.diagnosis instanceof Reference))
        throw new FHIRException("Type mismatch: the type Reference was expected, but "
            + this.diagnosis.getClass().getName() + " was encountered");
      return (Reference) this.diagnosis;
    }

    public boolean hasDiagnosisReference() {
        return this.diagnosis instanceof Reference;
    }

    public boolean hasDiagnosis() {
      return this.diagnosis != null && !this.diagnosis.isEmpty();
    }

    /**
     * @param value {@link #diagnosis} (The nature of illness or problem in a coded
     *              form or as a reference to an external defined Condition.)
     */
    public DiagnosisComponent setDiagnosis(DataType value) {
      if (value != null && !(value instanceof CodeableConcept || value instanceof Reference))
        throw new Error("Not the right type for ExplanationOfBenefit.diagnosis.diagnosis[x]: " + value.fhirType());
      this.diagnosis = value;
      return this;
    }

    /**
     * @return {@link #type} (When the condition was observed or the relative
     *         ranking.)
     */
    public List<CodeableConcept> getType() {
      if (this.type == null)
        this.type = new ArrayList<CodeableConcept>();
      return this.type;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DiagnosisComponent setType(List<CodeableConcept> theType) {
      this.type = theType;
      return this;
    }

    public boolean hasType() {
      if (this.type == null)
        return false;
      for (CodeableConcept item : this.type)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addType() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.type == null)
        this.type = new ArrayList<CodeableConcept>();
      this.type.add(t);
      return t;
    }

    public DiagnosisComponent addType(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.type == null)
        this.type = new ArrayList<CodeableConcept>();
      this.type.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #type}, creating it if
     *         it does not already exist {3}
     */
    public CodeableConcept getTypeFirstRep() {
      if (getType().isEmpty()) {
        addType();
      }
      return getType().get(0);
    }

    /**
     * @return {@link #onAdmission} (Indication of whether the diagnosis was present
     *         on admission to a facility.)
     */
    public CodeableConcept getOnAdmission() {
      if (this.onAdmission == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DiagnosisComponent.onAdmission");
        else if (Configuration.doAutoCreate())
          this.onAdmission = new CodeableConcept(); // cc
      return this.onAdmission;
    }

    public boolean hasOnAdmission() {
      return this.onAdmission != null && !this.onAdmission.isEmpty();
    }

    /**
     * @param value {@link #onAdmission} (Indication of whether the diagnosis was
     *              present on admission to a facility.)
     */
    public DiagnosisComponent setOnAdmission(CodeableConcept value) {
      this.onAdmission = value;
      return this;
    }

    /**
     * @return {@link #packageCode} (A package billing code or bundle code used to
     *         group products and services to a particular health condition (such as
     *         heart attack) which is based on a predetermined grouping code
     *         system.)
     */
    public CodeableConcept getPackageCode() {
      if (this.packageCode == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DiagnosisComponent.packageCode");
        else if (Configuration.doAutoCreate())
          this.packageCode = new CodeableConcept(); // cc
      return this.packageCode;
    }

    public boolean hasPackageCode() {
      return this.packageCode != null && !this.packageCode.isEmpty();
    }

    /**
     * @param value {@link #packageCode} (A package billing code or bundle code used
     *              to group products and services to a particular health condition
     *              (such as heart attack) which is based on a predetermined
     *              grouping code system.)
     */
    public DiagnosisComponent setPackageCode(CodeableConcept value) {
      this.packageCode = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(
          new Property("sequence", "positiveInt", "A number to uniquely identify diagnosis entries.", 0, 1, sequence));
      children.add(new Property("diagnosis[x]", "CodeableConcept|Reference(Condition)",
          "The nature of illness or problem in a coded form or as a reference to an external defined Condition.", 0, 1,
          diagnosis));
      children.add(new Property("type", "CodeableConcept", "When the condition was observed or the relative ranking.",
          0, java.lang.Integer.MAX_VALUE, type));
      children.add(new Property("onAdmission", "CodeableConcept",
          "Indication of whether the diagnosis was present on admission to a facility.", 0, 1, onAdmission));
      children.add(new Property("packageCode", "CodeableConcept",
          "A package billing code or bundle code used to group products and services to a particular health condition (such as heart attack) which is based on a predetermined grouping code system.",
          0, 1, packageCode));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt",
            "A number to uniquely identify diagnosis entries.", 0, 1, sequence);
      case -1487009809:
        /* diagnosis[x] */ return new Property("diagnosis[x]", "CodeableConcept|Reference(Condition)",
            "The nature of illness or problem in a coded form or as a reference to an external defined Condition.", 0,
            1, diagnosis);
      case 1196993265:
        /* diagnosis */ return new Property("diagnosis[x]", "CodeableConcept|Reference(Condition)",
            "The nature of illness or problem in a coded form or as a reference to an external defined Condition.", 0,
            1, diagnosis);
      case 277781616:
        /* diagnosisCodeableConcept */ return new Property("diagnosis[x]", "CodeableConcept",
            "The nature of illness or problem in a coded form or as a reference to an external defined Condition.", 0,
            1, diagnosis);
      case 2050454362:
        /* diagnosisReference */ return new Property("diagnosis[x]", "Reference(Condition)",
            "The nature of illness or problem in a coded form or as a reference to an external defined Condition.", 0,
            1, diagnosis);
      case 3575610:
        /* type */ return new Property("type", "CodeableConcept",
            "When the condition was observed or the relative ranking.", 0, java.lang.Integer.MAX_VALUE, type);
      case -3386134:
        /* onAdmission */ return new Property("onAdmission", "CodeableConcept",
            "Indication of whether the diagnosis was present on admission to a facility.", 0, 1, onAdmission);
      case 908444499:
        /* packageCode */ return new Property("packageCode", "CodeableConcept",
            "A package billing code or bundle code used to group products and services to a particular health condition (such as heart attack) which is based on a predetermined grouping code system.",
            0, 1, packageCode);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case 1196993265:
        /* diagnosis */ return this.diagnosis == null ? new Base[0] : new Base[] { this.diagnosis }; // DataType
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : this.type.toArray(new Base[this.type.size()]); // CodeableConcept
      case -3386134:
        /* onAdmission */ return this.onAdmission == null ? new Base[0] : new Base[] { this.onAdmission }; // CodeableConcept
      case 908444499:
        /* packageCode */ return this.packageCode == null ? new Base[0] : new Base[] { this.packageCode }; // CodeableConcept
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 1196993265: // diagnosis
        this.diagnosis = TypeConvertor.castToType(value); // DataType
        return value;
      case 3575610: // type
        this.getType().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case -3386134: // onAdmission
        this.onAdmission = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 908444499: // packageCode
        this.packageCode = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("diagnosis[x]")) {
        this.diagnosis = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("type")) {
        this.getType().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("onAdmission")) {
        this.onAdmission = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("packageCode")) {
        this.packageCode = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("diagnosis[x]")) {
        this.diagnosis = null;
      } else if (name.equals("type")) {
        this.getType().remove(value);
      } else if (name.equals("onAdmission")) {
        this.onAdmission = null;
      } else if (name.equals("packageCode")) {
        this.packageCode = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case -1487009809:
        return getDiagnosis();
      case 1196993265:
        return getDiagnosis();
      case 3575610:
        return addType();
      case -3386134:
        return getOnAdmission();
      case 908444499:
        return getPackageCode();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case 1196993265:
        /* diagnosis */ return new String[] { "CodeableConcept", "Reference" };
      case 3575610:
        /* type */ return new String[] { "CodeableConcept" };
      case -3386134:
        /* onAdmission */ return new String[] { "CodeableConcept" };
      case 908444499:
        /* packageCode */ return new String[] { "CodeableConcept" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.diagnosis.sequence");
      } else if (name.equals("diagnosisCodeableConcept")) {
        this.diagnosis = new CodeableConcept();
        return this.diagnosis;
      } else if (name.equals("diagnosisReference")) {
        this.diagnosis = new Reference();
        return this.diagnosis;
      } else if (name.equals("type")) {
        return addType();
      } else if (name.equals("onAdmission")) {
        this.onAdmission = new CodeableConcept();
        return this.onAdmission;
      } else if (name.equals("packageCode")) {
        this.packageCode = new CodeableConcept();
        return this.packageCode;
      } else
        return super.addChild(name);
    }

    public DiagnosisComponent copy() {
      DiagnosisComponent dst = new DiagnosisComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(DiagnosisComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      dst.diagnosis = diagnosis == null ? null : diagnosis.copy();
      if (type != null) {
        dst.type = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : type)
          dst.type.add(i.copy());
      }
      ;
      dst.onAdmission = onAdmission == null ? null : onAdmission.copy();
      dst.packageCode = packageCode == null ? null : packageCode.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof DiagnosisComponent))
        return false;
      DiagnosisComponent o = (DiagnosisComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(diagnosis, o.diagnosis, true)
          && compareDeep(type, o.type, true) && compareDeep(onAdmission, o.onAdmission, true)
          && compareDeep(packageCode, o.packageCode, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof DiagnosisComponent))
        return false;
      DiagnosisComponent o = (DiagnosisComponent) other_;
      return compareValues(sequence, o.sequence, true);
    }

    public boolean isEmpty() {
      return super.isEmpty()
          && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, diagnosis, type, onAdmission, packageCode);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.diagnosis";

    }

  }

  @Block()
  public static class ProcedureComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A number to uniquely identify procedure entries.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Procedure instance identifier", formalDefinition = "A number to uniquely identify procedure entries.")
    protected PositiveIntType sequence;

    /**
     * When the condition was observed or the relative ranking.
     */
    @Child(name = "type", type = {
        CodeableConcept.class }, order = 2, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Category of Procedure", formalDefinition = "When the condition was observed or the relative ranking.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-procedure-type")
    protected List<CodeableConcept> type;

    /**
     * Date and optionally time the procedure was performed.
     */
    @Child(name = "date", type = { DateTimeType.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "When the procedure was performed", formalDefinition = "Date and optionally time the procedure was performed.")
    protected DateTimeType date;

    /**
     * The code or reference to a Procedure resource which identifies the clinical
     * intervention performed.
     */
    @Child(name = "procedure", type = { CodeableConcept.class,
        Procedure.class }, order = 4, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Specific clinical procedure", formalDefinition = "The code or reference to a Procedure resource which identifies the clinical intervention performed.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/icd-10-procedures")
    protected DataType procedure;

    /**
     * Unique Device Identifiers associated with this line item.
     */
    @Child(name = "udi", type = {
        Device.class }, order = 5, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Unique device identifier", formalDefinition = "Unique Device Identifiers associated with this line item.")
    protected List<Reference> udi;

    private static final long serialVersionUID = 1165684715L;

    /**
     * Constructor
     */
    public ProcedureComponent() {
      super();
    }

    /**
     * Constructor
     */
    public ProcedureComponent(int sequence, DataType procedure) {
      super();
      this.setSequence(sequence);
      this.setProcedure(procedure);
    }

    /**
     * @return {@link #sequence} (A number to uniquely identify procedure entries.).
     *         This is the underlying object with id, value and extensions. The
     *         accessor "getSequence" gives direct access to the value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ProcedureComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A number to uniquely identify procedure
     *              entries.). This is the underlying object with id, value and
     *              extensions. The accessor "getSequence" gives direct access to
     *              the value
     */
    public ProcedureComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A number to uniquely identify procedure entries.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A number to uniquely identify procedure entries.
     */
    public ProcedureComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #type} (When the condition was observed or the relative
     *         ranking.)
     */
    public List<CodeableConcept> getType() {
      if (this.type == null)
        this.type = new ArrayList<CodeableConcept>();
      return this.type;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ProcedureComponent setType(List<CodeableConcept> theType) {
      this.type = theType;
      return this;
    }

    public boolean hasType() {
      if (this.type == null)
        return false;
      for (CodeableConcept item : this.type)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addType() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.type == null)
        this.type = new ArrayList<CodeableConcept>();
      this.type.add(t);
      return t;
    }

    public ProcedureComponent addType(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.type == null)
        this.type = new ArrayList<CodeableConcept>();
      this.type.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #type}, creating it if
     *         it does not already exist {3}
     */
    public CodeableConcept getTypeFirstRep() {
      if (getType().isEmpty()) {
        addType();
      }
      return getType().get(0);
    }

    /**
     * @return {@link #date} (Date and optionally time the procedure was
     *         performed.). This is the underlying object with id, value and
     *         extensions. The accessor "getDate" gives direct access to the value
     */
    public DateTimeType getDateElement() {
      if (this.date == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ProcedureComponent.date");
        else if (Configuration.doAutoCreate())
          this.date = new DateTimeType(); // bb
      return this.date;
    }

    public boolean hasDateElement() {
      return this.date != null && !this.date.isEmpty();
    }

    public boolean hasDate() {
      return this.date != null && !this.date.isEmpty();
    }

    /**
     * @param value {@link #date} (Date and optionally time the procedure was
     *              performed.). This is the underlying object with id, value and
     *              extensions. The accessor "getDate" gives direct access to the
     *              value
     */
    public ProcedureComponent setDateElement(DateTimeType value) {
      this.date = value;
      return this;
    }

    /**
     * @return Date and optionally time the procedure was performed.
     */
    public Date getDate() {
      return this.date == null ? null : this.date.getValue();
    }

    /**
     * @param value Date and optionally time the procedure was performed.
     */
    public ProcedureComponent setDate(Date value) {
      if (value == null)
        this.date = null;
      else {
        if (this.date == null)
          this.date = new DateTimeType();
        this.date.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #procedure} (The code or reference to a Procedure resource
     *         which identifies the clinical intervention performed.)
     */
    public DataType getProcedure() {
      return this.procedure;
    }

    /**
     * @return {@link #procedure} (The code or reference to a Procedure resource
     *         which identifies the clinical intervention performed.)
     */
    public CodeableConcept getProcedureCodeableConcept() throws FHIRException {
      if (this.procedure == null)
        this.procedure = new CodeableConcept();
      if (!(this.procedure instanceof CodeableConcept))
        throw new FHIRException("Type mismatch: the type CodeableConcept was expected, but "
            + this.procedure.getClass().getName() + " was encountered");
      return (CodeableConcept) this.procedure;
    }

    public boolean hasProcedureCodeableConcept() {
        return this.procedure instanceof CodeableConcept;
    }

    /**
     * @return {@link #procedure} (The code or reference to a Procedure resource
     *         which identifies the clinical intervention performed.)
     */
    public Reference getProcedureReference() throws FHIRException {
      if (this.procedure == null)
        this.procedure = new Reference();
      if (!(this.procedure instanceof Reference))
        throw new FHIRException("Type mismatch: the type Reference was expected, but "
            + this.procedure.getClass().getName() + " was encountered");
      return (Reference) this.procedure;
    }

    public boolean hasProcedureReference() {
        return this.procedure instanceof Reference;
    }

    public boolean hasProcedure() {
      return this.procedure != null && !this.procedure.isEmpty();
    }

    /**
     * @param value {@link #procedure} (The code or reference to a Procedure
     *              resource which identifies the clinical intervention performed.)
     */
    public ProcedureComponent setProcedure(DataType value) {
      if (value != null && !(value instanceof CodeableConcept || value instanceof Reference))
        throw new Error("Not the right type for ExplanationOfBenefit.procedure.procedure[x]: " + value.fhirType());
      this.procedure = value;
      return this;
    }

    /**
     * @return {@link #udi} (Unique Device Identifiers associated with this line
     *         item.)
     */
    public List<Reference> getUdi() {
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      return this.udi;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ProcedureComponent setUdi(List<Reference> theUdi) {
      this.udi = theUdi;
      return this;
    }

    public boolean hasUdi() {
      if (this.udi == null)
        return false;
      for (Reference item : this.udi)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public Reference addUdi() { // 3
      Reference t = new Reference();
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return t;
    }

    public ProcedureComponent addUdi(Reference t) { // 3
      if (t == null)
        return this;
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #udi}, creating it if
     *         it does not already exist {3}
     */
    public Reference getUdiFirstRep() {
      if (getUdi().isEmpty()) {
        addUdi();
      }
      return getUdi().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(
          new Property("sequence", "positiveInt", "A number to uniquely identify procedure entries.", 0, 1, sequence));
      children.add(new Property("type", "CodeableConcept", "When the condition was observed or the relative ranking.",
          0, java.lang.Integer.MAX_VALUE, type));
      children
          .add(new Property("date", "dateTime", "Date and optionally time the procedure was performed.", 0, 1, date));
      children.add(new Property("procedure[x]", "CodeableConcept|Reference(Procedure)",
          "The code or reference to a Procedure resource which identifies the clinical intervention performed.", 0, 1,
          procedure));
      children.add(new Property("udi", "Reference(Device)", "Unique Device Identifiers associated with this line item.",
          0, java.lang.Integer.MAX_VALUE, udi));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt",
            "A number to uniquely identify procedure entries.", 0, 1, sequence);
      case 3575610:
        /* type */ return new Property("type", "CodeableConcept",
            "When the condition was observed or the relative ranking.", 0, java.lang.Integer.MAX_VALUE, type);
      case 3076014:
        /* date */ return new Property("date", "dateTime", "Date and optionally time the procedure was performed.", 0,
            1, date);
      case 1640074445:
        /* procedure[x] */ return new Property("procedure[x]", "CodeableConcept|Reference(Procedure)",
            "The code or reference to a Procedure resource which identifies the clinical intervention performed.", 0, 1,
            procedure);
      case -1095204141:
        /* procedure */ return new Property("procedure[x]", "CodeableConcept|Reference(Procedure)",
            "The code or reference to a Procedure resource which identifies the clinical intervention performed.", 0, 1,
            procedure);
      case -1284783026:
        /* procedureCodeableConcept */ return new Property("procedure[x]", "CodeableConcept",
            "The code or reference to a Procedure resource which identifies the clinical intervention performed.", 0, 1,
            procedure);
      case 881809848:
        /* procedureReference */ return new Property("procedure[x]", "Reference(Procedure)",
            "The code or reference to a Procedure resource which identifies the clinical intervention performed.", 0, 1,
            procedure);
      case 115642:
        /* udi */ return new Property("udi", "Reference(Device)",
            "Unique Device Identifiers associated with this line item.", 0, java.lang.Integer.MAX_VALUE, udi);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : this.type.toArray(new Base[this.type.size()]); // CodeableConcept
      case 3076014:
        /* date */ return this.date == null ? new Base[0] : new Base[] { this.date }; // DateTimeType
      case -1095204141:
        /* procedure */ return this.procedure == null ? new Base[0] : new Base[] { this.procedure }; // DataType
      case 115642:
        /* udi */ return this.udi == null ? new Base[0] : this.udi.toArray(new Base[this.udi.size()]); // Reference
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 3575610: // type
        this.getType().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 3076014: // date
        this.date = TypeConvertor.castToDateTime(value); // DateTimeType
        return value;
      case -1095204141: // procedure
        this.procedure = TypeConvertor.castToType(value); // DataType
        return value;
      case 115642: // udi
        this.getUdi().add(TypeConvertor.castToReference(value)); // Reference
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("type")) {
        this.getType().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("date")) {
        this.date = TypeConvertor.castToDateTime(value); // DateTimeType
      } else if (name.equals("procedure[x]")) {
        this.procedure = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("udi")) {
        this.getUdi().add(TypeConvertor.castToReference(value));
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("type")) {
        this.getType().remove(value);
      } else if (name.equals("date")) {
        this.date = null;
      } else if (name.equals("procedure[x]")) {
        this.procedure = null;
      } else if (name.equals("udi")) {
        this.getUdi().remove(value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case 3575610:
        return addType();
      case 3076014:
        return getDateElement();
      case 1640074445:
        return getProcedure();
      case -1095204141:
        return getProcedure();
      case 115642:
        return addUdi();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case 3575610:
        /* type */ return new String[] { "CodeableConcept" };
      case 3076014:
        /* date */ return new String[] { "dateTime" };
      case -1095204141:
        /* procedure */ return new String[] { "CodeableConcept", "Reference" };
      case 115642:
        /* udi */ return new String[] { "Reference" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.procedure.sequence");
      } else if (name.equals("type")) {
        return addType();
      } else if (name.equals("date")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.procedure.date");
      } else if (name.equals("procedureCodeableConcept")) {
        this.procedure = new CodeableConcept();
        return this.procedure;
      } else if (name.equals("procedureReference")) {
        this.procedure = new Reference();
        return this.procedure;
      } else if (name.equals("udi")) {
        return addUdi();
      } else
        return super.addChild(name);
    }

    public ProcedureComponent copy() {
      ProcedureComponent dst = new ProcedureComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(ProcedureComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      if (type != null) {
        dst.type = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : type)
          dst.type.add(i.copy());
      }
      ;
      dst.date = date == null ? null : date.copy();
      dst.procedure = procedure == null ? null : procedure.copy();
      if (udi != null) {
        dst.udi = new ArrayList<Reference>();
        for (Reference i : udi)
          dst.udi.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof ProcedureComponent))
        return false;
      ProcedureComponent o = (ProcedureComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(type, o.type, true)
          && compareDeep(date, o.date, true) && compareDeep(procedure, o.procedure, true)
          && compareDeep(udi, o.udi, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof ProcedureComponent))
        return false;
      ProcedureComponent o = (ProcedureComponent) other_;
      return compareValues(sequence, o.sequence, true) && compareValues(date, o.date, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, type, date, procedure, udi);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.procedure";

    }

  }

  @Block()
  public static class InsuranceComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A flag to indicate that this Coverage is to be used for adjudication of this
     * claim when set to true.
     */
    @Child(name = "focal", type = { BooleanType.class }, order = 1, min = 1, max = 1, modifier = false, summary = true)
    @Description(shortDefinition = "Coverage to be used for adjudication", formalDefinition = "A flag to indicate that this Coverage is to be used for adjudication of this claim when set to true.")
    protected BooleanType focal;

    /**
     * Reference to the insurance card level information contained in the Coverage
     * resource. The coverage issuing insurer will use these details to locate the
     * patient's actual coverage within the insurer's information system.
     */
    @Child(name = "coverage", type = { Coverage.class }, order = 2, min = 1, max = 1, modifier = false, summary = true)
    @Description(shortDefinition = "Insurance information", formalDefinition = "Reference to the insurance card level information contained in the Coverage resource. The coverage issuing insurer will use these details to locate the patient's actual coverage within the insurer's information system.")
    protected Reference coverage;

    /**
     * Reference numbers previously provided by the insurer to the provider to be
     * quoted on subsequent claims containing services or products related to the
     * prior authorization.
     */
    @Child(name = "preAuthRef", type = {
        StringType.class }, order = 3, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Prior authorization reference number", formalDefinition = "Reference numbers previously provided by the insurer to the provider to be quoted on subsequent claims containing services or products related to the prior authorization.")
    protected List<StringType> preAuthRef;

    private static final long serialVersionUID = 1519900285L;

    /**
     * Constructor
     */
    public InsuranceComponent() {
      super();
    }

    /**
     * Constructor
     */
    public InsuranceComponent(boolean focal, Reference coverage) {
      super();
      this.setFocal(focal);
      this.setCoverage(coverage);
    }

    /**
     * @return {@link #focal} (A flag to indicate that this Coverage is to be used
     *         for adjudication of this claim when set to true.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFocal" gives direct access to the value
     */
    public BooleanType getFocalElement() {
      if (this.focal == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create InsuranceComponent.focal");
        else if (Configuration.doAutoCreate())
          this.focal = new BooleanType(); // bb
      return this.focal;
    }

    public boolean hasFocalElement() {
      return this.focal != null && !this.focal.isEmpty();
    }

    public boolean hasFocal() {
      return this.focal != null && !this.focal.isEmpty();
    }

    /**
     * @param value {@link #focal} (A flag to indicate that this Coverage is to be
     *              used for adjudication of this claim when set to true.). This is
     *              the underlying object with id, value and extensions. The
     *              accessor "getFocal" gives direct access to the value
     */
    public InsuranceComponent setFocalElement(BooleanType value) {
      this.focal = value;
      return this;
    }

    /**
     * @return A flag to indicate that this Coverage is to be used for adjudication
     *         of this claim when set to true.
     */
    public boolean getFocal() {
      return this.focal == null || this.focal.isEmpty() ? false : this.focal.getValue();
    }

    /**
     * @param value A flag to indicate that this Coverage is to be used for
     *              adjudication of this claim when set to true.
     */
    public InsuranceComponent setFocal(boolean value) {
      if (this.focal == null)
        this.focal = new BooleanType();
      this.focal.setValue(value);
      return this;
    }

    /**
     * @return {@link #coverage} (Reference to the insurance card level information
     *         contained in the Coverage resource. The coverage issuing insurer will
     *         use these details to locate the patient's actual coverage within the
     *         insurer's information system.)
     */
    public Reference getCoverage() {
      if (this.coverage == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create InsuranceComponent.coverage");
        else if (Configuration.doAutoCreate())
          this.coverage = new Reference(); // cc
      return this.coverage;
    }

    public boolean hasCoverage() {
      return this.coverage != null && !this.coverage.isEmpty();
    }

    /**
     * @param value {@link #coverage} (Reference to the insurance card level
     *              information contained in the Coverage resource. The coverage
     *              issuing insurer will use these details to locate the patient's
     *              actual coverage within the insurer's information system.)
     */
    public InsuranceComponent setCoverage(Reference value) {
      this.coverage = value;
      return this;
    }

    /**
     * @return {@link #preAuthRef} (Reference numbers previously provided by the
     *         insurer to the provider to be quoted on subsequent claims containing
     *         services or products related to the prior authorization.)
     */
    public List<StringType> getPreAuthRef() {
      if (this.preAuthRef == null)
        this.preAuthRef = new ArrayList<StringType>();
      return this.preAuthRef;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public InsuranceComponent setPreAuthRef(List<StringType> thePreAuthRef) {
      this.preAuthRef = thePreAuthRef;
      return this;
    }

    public boolean hasPreAuthRef() {
      if (this.preAuthRef == null)
        return false;
      for (StringType item : this.preAuthRef)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #preAuthRef} (Reference numbers previously provided by the
     *         insurer to the provider to be quoted on subsequent claims containing
     *         services or products related to the prior authorization.)
     */
    public StringType addPreAuthRefElement() {// 2
      StringType t = new StringType();
      if (this.preAuthRef == null)
        this.preAuthRef = new ArrayList<StringType>();
      this.preAuthRef.add(t);
      return t;
    }

    /**
     * @param value {@link #preAuthRef} (Reference numbers previously provided by
     *              the insurer to the provider to be quoted on subsequent claims
     *              containing services or products related to the prior
     *              authorization.)
     */
    public InsuranceComponent addPreAuthRef(String value) { // 1
      StringType t = new StringType();
      t.setValue(value);
      if (this.preAuthRef == null)
        this.preAuthRef = new ArrayList<StringType>();
      this.preAuthRef.add(t);
      return this;
    }

    /**
     * @param value {@link #preAuthRef} (Reference numbers previously provided by
     *              the insurer to the provider to be quoted on subsequent claims
     *              containing services or products related to the prior
     *              authorization.)
     */
    public boolean hasPreAuthRef(String value) {
      if (this.preAuthRef == null)
        return false;
      for (StringType v : this.preAuthRef)
        if (v.getValue().equals(value)) // string
          return true;
      return false;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("focal", "boolean",
          "A flag to indicate that this Coverage is to be used for adjudication of this claim when set to true.", 0, 1,
          focal));
      children.add(new Property("coverage", "Reference(Coverage)",
          "Reference to the insurance card level information contained in the Coverage resource. The coverage issuing insurer will use these details to locate the patient's actual coverage within the insurer's information system.",
          0, 1, coverage));
      children.add(new Property("preAuthRef", "string",
          "Reference numbers previously provided by the insurer to the provider to be quoted on subsequent claims containing services or products related to the prior authorization.",
          0, java.lang.Integer.MAX_VALUE, preAuthRef));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 97604197:
        /* focal */ return new Property("focal", "boolean",
            "A flag to indicate that this Coverage is to be used for adjudication of this claim when set to true.", 0,
            1, focal);
      case -351767064:
        /* coverage */ return new Property("coverage", "Reference(Coverage)",
            "Reference to the insurance card level information contained in the Coverage resource. The coverage issuing insurer will use these details to locate the patient's actual coverage within the insurer's information system.",
            0, 1, coverage);
      case 522246568:
        /* preAuthRef */ return new Property("preAuthRef", "string",
            "Reference numbers previously provided by the insurer to the provider to be quoted on subsequent claims containing services or products related to the prior authorization.",
            0, java.lang.Integer.MAX_VALUE, preAuthRef);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 97604197:
        /* focal */ return this.focal == null ? new Base[0] : new Base[] { this.focal }; // BooleanType
      case -351767064:
        /* coverage */ return this.coverage == null ? new Base[0] : new Base[] { this.coverage }; // Reference
      case 522246568:
        /* preAuthRef */ return this.preAuthRef == null ? new Base[0]
            : this.preAuthRef.toArray(new Base[this.preAuthRef.size()]); // StringType
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 97604197: // focal
        this.focal = TypeConvertor.castToBoolean(value); // BooleanType
        return value;
      case -351767064: // coverage
        this.coverage = TypeConvertor.castToReference(value); // Reference
        return value;
      case 522246568: // preAuthRef
        this.getPreAuthRef().add(TypeConvertor.castToString(value)); // StringType
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("focal")) {
        this.focal = TypeConvertor.castToBoolean(value); // BooleanType
      } else if (name.equals("coverage")) {
        this.coverage = TypeConvertor.castToReference(value); // Reference
      } else if (name.equals("preAuthRef")) {
        this.getPreAuthRef().add(TypeConvertor.castToString(value));
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("focal")) {
        this.focal = null;
      } else if (name.equals("coverage")) {
        this.coverage = null;
      } else if (name.equals("preAuthRef")) {
        this.getPreAuthRef().remove(value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 97604197:
        return getFocalElement();
      case -351767064:
        return getCoverage();
      case 522246568:
        return addPreAuthRefElement();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 97604197:
        /* focal */ return new String[] { "boolean" };
      case -351767064:
        /* coverage */ return new String[] { "Reference" };
      case 522246568:
        /* preAuthRef */ return new String[] { "string" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("focal")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.insurance.focal");
      } else if (name.equals("coverage")) {
        this.coverage = new Reference();
        return this.coverage;
      } else if (name.equals("preAuthRef")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.insurance.preAuthRef");
      } else
        return super.addChild(name);
    }

    public InsuranceComponent copy() {
      InsuranceComponent dst = new InsuranceComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(InsuranceComponent dst) {
      super.copyValues(dst);
      dst.focal = focal == null ? null : focal.copy();
      dst.coverage = coverage == null ? null : coverage.copy();
      if (preAuthRef != null) {
        dst.preAuthRef = new ArrayList<StringType>();
        for (StringType i : preAuthRef)
          dst.preAuthRef.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof InsuranceComponent))
        return false;
      InsuranceComponent o = (InsuranceComponent) other_;
      return compareDeep(focal, o.focal, true) && compareDeep(coverage, o.coverage, true)
          && compareDeep(preAuthRef, o.preAuthRef, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof InsuranceComponent))
        return false;
      InsuranceComponent o = (InsuranceComponent) other_;
      return compareValues(focal, o.focal, true) && compareValues(preAuthRef, o.preAuthRef, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(focal, coverage, preAuthRef);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.insurance";

    }

  }

  @Block()
  public static class AccidentComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Date of an accident event related to the products and services contained in
     * the claim.
     */
    @Child(name = "date", type = { DateType.class }, order = 1, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "When the incident occurred", formalDefinition = "Date of an accident event  related to the products and services contained in the claim.")
    protected DateType date;

    /**
     * The type or context of the accident event for the purposes of selection of
     * potential insurance coverages and determination of coordination between
     * insurers.
     */
    @Child(name = "type", type = {
        CodeableConcept.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "The nature of the accident", formalDefinition = "The type or context of the accident event for the purposes of selection of potential insurance coverages and determination of coordination between insurers.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://terminology.hl7.org/ValueSet/v3-ActIncidentCode")
    protected CodeableConcept type;

    /**
     * The physical location of the accident event.
     */
    @Child(name = "location", type = { Address.class,
        Location.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Where the event occurred", formalDefinition = "The physical location of the accident event.")
    protected DataType location;

    private static final long serialVersionUID = 11882722L;

    /**
     * Constructor
     */
    public AccidentComponent() {
      super();
    }

    /**
     * @return {@link #date} (Date of an accident event related to the products and
     *         services contained in the claim.). This is the underlying object with
     *         id, value and extensions. The accessor "getDate" gives direct access
     *         to the value
     */
    public DateType getDateElement() {
      if (this.date == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AccidentComponent.date");
        else if (Configuration.doAutoCreate())
          this.date = new DateType(); // bb
      return this.date;
    }

    public boolean hasDateElement() {
      return this.date != null && !this.date.isEmpty();
    }

    public boolean hasDate() {
      return this.date != null && !this.date.isEmpty();
    }

    /**
     * @param value {@link #date} (Date of an accident event related to the products
     *              and services contained in the claim.). This is the underlying
     *              object with id, value and extensions. The accessor "getDate"
     *              gives direct access to the value
     */
    public AccidentComponent setDateElement(DateType value) {
      this.date = value;
      return this;
    }

    /**
     * @return Date of an accident event related to the products and services
     *         contained in the claim.
     */
    public Date getDate() {
      return this.date == null ? null : this.date.getValue();
    }

    /**
     * @param value Date of an accident event related to the products and services
     *              contained in the claim.
     */
    public AccidentComponent setDate(Date value) {
      if (value == null)
        this.date = null;
      else {
        if (this.date == null)
          this.date = new DateType();
        this.date.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #type} (The type or context of the accident event for the
     *         purposes of selection of potential insurance coverages and
     *         determination of coordination between insurers.)
     */
    public CodeableConcept getType() {
      if (this.type == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AccidentComponent.type");
        else if (Configuration.doAutoCreate())
          this.type = new CodeableConcept(); // cc
      return this.type;
    }

    public boolean hasType() {
      return this.type != null && !this.type.isEmpty();
    }

    /**
     * @param value {@link #type} (The type or context of the accident event for the
     *              purposes of selection of potential insurance coverages and
     *              determination of coordination between insurers.)
     */
    public AccidentComponent setType(CodeableConcept value) {
      this.type = value;
      return this;
    }

    /**
     * @return {@link #location} (The physical location of the accident event.)
     */
    public DataType getLocation() {
      return this.location;
    }

    /**
     * @return {@link #location} (The physical location of the accident event.)
     */
    public Address getLocationAddress() throws FHIRException {
      if (this.location == null)
        this.location = new Address();
      if (!(this.location instanceof Address))
        throw new FHIRException("Type mismatch: the type Address was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (Address) this.location;
    }

    public boolean hasLocationAddress() {
        return this.location instanceof Address;
    }

    /**
     * @return {@link #location} (The physical location of the accident event.)
     */
    public Reference getLocationReference() throws FHIRException {
      if (this.location == null)
        this.location = new Reference();
      if (!(this.location instanceof Reference))
        throw new FHIRException("Type mismatch: the type Reference was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (Reference) this.location;
    }

    public boolean hasLocationReference() {
        return this.location instanceof Reference;
    }

    public boolean hasLocation() {
      return this.location != null && !this.location.isEmpty();
    }

    /**
     * @param value {@link #location} (The physical location of the accident event.)
     */
    public AccidentComponent setLocation(DataType value) {
      if (value != null && !(value instanceof Address || value instanceof Reference))
        throw new Error("Not the right type for ExplanationOfBenefit.accident.location[x]: " + value.fhirType());
      this.location = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("date", "date",
          "Date of an accident event  related to the products and services contained in the claim.", 0, 1, date));
      children.add(new Property("type", "CodeableConcept",
          "The type or context of the accident event for the purposes of selection of potential insurance coverages and determination of coordination between insurers.",
          0, 1, type));
      children.add(new Property("location[x]", "Address|Reference(Location)",
          "The physical location of the accident event.", 0, 1, location));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 3076014:
        /* date */ return new Property("date", "date",
            "Date of an accident event  related to the products and services contained in the claim.", 0, 1, date);
      case 3575610:
        /* type */ return new Property("type", "CodeableConcept",
            "The type or context of the accident event for the purposes of selection of potential insurance coverages and determination of coordination between insurers.",
            0, 1, type);
      case 552316075:
        /* location[x] */ return new Property("location[x]", "Address|Reference(Location)",
            "The physical location of the accident event.", 0, 1, location);
      case 1901043637:
        /* location */ return new Property("location[x]", "Address|Reference(Location)",
            "The physical location of the accident event.", 0, 1, location);
      case -1280020865:
        /* locationAddress */ return new Property("location[x]", "Address",
            "The physical location of the accident event.", 0, 1, location);
      case 755866390:
        /* locationReference */ return new Property("location[x]", "Reference(Location)",
            "The physical location of the accident event.", 0, 1, location);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 3076014:
        /* date */ return this.date == null ? new Base[0] : new Base[] { this.date }; // DateType
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : new Base[] { this.type }; // CodeableConcept
      case 1901043637:
        /* location */ return this.location == null ? new Base[0] : new Base[] { this.location }; // DataType
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 3076014: // date
        this.date = TypeConvertor.castToDate(value); // DateType
        return value;
      case 3575610: // type
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 1901043637: // location
        this.location = TypeConvertor.castToType(value); // DataType
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("date")) {
        this.date = TypeConvertor.castToDate(value); // DateType
      } else if (name.equals("type")) {
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("location[x]")) {
        this.location = TypeConvertor.castToType(value); // DataType
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("date")) {
        this.date = null;
      } else if (name.equals("type")) {
        this.type = null;
      } else if (name.equals("location[x]")) {
        this.location = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3076014:
        return getDateElement();
      case 3575610:
        return getType();
      case 552316075:
        return getLocation();
      case 1901043637:
        return getLocation();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3076014:
        /* date */ return new String[] { "date" };
      case 3575610:
        /* type */ return new String[] { "CodeableConcept" };
      case 1901043637:
        /* location */ return new String[] { "Address", "Reference" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("date")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.accident.date");
      } else if (name.equals("type")) {
        this.type = new CodeableConcept();
        return this.type;
      } else if (name.equals("locationAddress")) {
        this.location = new Address();
        return this.location;
      } else if (name.equals("locationReference")) {
        this.location = new Reference();
        return this.location;
      } else
        return super.addChild(name);
    }

    public AccidentComponent copy() {
      AccidentComponent dst = new AccidentComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(AccidentComponent dst) {
      super.copyValues(dst);
      dst.date = date == null ? null : date.copy();
      dst.type = type == null ? null : type.copy();
      dst.location = location == null ? null : location.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof AccidentComponent))
        return false;
      AccidentComponent o = (AccidentComponent) other_;
      return compareDeep(date, o.date, true) && compareDeep(type, o.type, true)
          && compareDeep(location, o.location, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof AccidentComponent))
        return false;
      AccidentComponent o = (AccidentComponent) other_;
      return compareValues(date, o.date, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(date, type, location);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.accident";

    }

  }

  @Block()
  public static class ItemComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A number to uniquely identify item entries.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Item instance identifier", formalDefinition = "A number to uniquely identify item entries.")
    protected PositiveIntType sequence;

    /**
     * Care team members related to this service or product.
     */
    @Child(name = "careTeamSequence", type = {
        PositiveIntType.class }, order = 2, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable care team members", formalDefinition = "Care team members related to this service or product.")
    protected List<PositiveIntType> careTeamSequence;

    /**
     * Diagnoses applicable for this service or product.
     */
    @Child(name = "diagnosisSequence", type = {
        PositiveIntType.class }, order = 3, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable diagnoses", formalDefinition = "Diagnoses applicable for this service or product.")
    protected List<PositiveIntType> diagnosisSequence;

    /**
     * Procedures applicable for this service or product.
     */
    @Child(name = "procedureSequence", type = {
        PositiveIntType.class }, order = 4, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable procedures", formalDefinition = "Procedures applicable for this service or product.")
    protected List<PositiveIntType> procedureSequence;

    /**
     * Exceptions, special conditions and supporting information applicable for this
     * service or product.
     */
    @Child(name = "informationSequence", type = {
        PositiveIntType.class }, order = 5, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable exception and supporting information", formalDefinition = "Exceptions, special conditions and supporting information applicable for this service or product.")
    protected List<PositiveIntType> informationSequence;

    /**
     * The type of revenue or cost center providing the product and/or service.
     */
    @Child(name = "revenue", type = {
        CodeableConcept.class }, order = 6, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Revenue or cost center code", formalDefinition = "The type of revenue or cost center providing the product and/or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-revenue-center")
    protected CodeableConcept revenue;

    /**
     * Code to identify the general type of benefits under which products and
     * services are provided.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 7, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefit classification", formalDefinition = "Code to identify the general type of benefits under which products and services are provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-benefitcategory")
    protected CodeableConcept category;

    /**
     * When the value is a group code then this item collects a set of related claim
     * details, otherwise this contains the product, service, drug or other billing
     * code for the item.
     */
    @Child(name = "productOrService", type = {
        CodeableConcept.class }, order = 8, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Billing, service, product, or drug code", formalDefinition = "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-uscls")
    protected CodeableConcept productOrService;

    /**
     * Item typification or modifiers codes to convey additional context for the
     * product or service.
     */
    @Child(name = "modifier", type = {
        CodeableConcept.class }, order = 9, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Product or service billing modifiers", formalDefinition = "Item typification or modifiers codes to convey additional context for the product or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-modifiers")
    protected List<CodeableConcept> modifier;

    /**
     * Identifies the program under which this may be recovered.
     */
    @Child(name = "programCode", type = {
        CodeableConcept.class }, order = 10, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Program the product or service is provided under", formalDefinition = "Identifies the program under which this may be recovered.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-program-code")
    protected List<CodeableConcept> programCode;

    /**
     * The date or dates when the service or product was supplied, performed or
     * completed.
     */
    @Child(name = "serviced", type = { DateType.class,
        Period.class }, order = 11, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Date or dates of service or product delivery", formalDefinition = "The date or dates when the service or product was supplied, performed or completed.")
    protected DataType serviced;

    /**
     * Where the product or service was provided.
     */
    @Child(name = "location", type = { CodeableConcept.class, Address.class,
        Location.class }, order = 12, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Place of service or where product was supplied", formalDefinition = "Where the product or service was provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-place")
    protected DataType location;

    /**
     * The number of repetitions of a service or product.
     */
    @Child(name = "quantity", type = {
        Quantity.class }, order = 13, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Count of products or services", formalDefinition = "The number of repetitions of a service or product.")
    protected Quantity quantity;

    /**
     * If the item is not a group then this is the fee for the product or service,
     * otherwise this is the total of the fees for the details of the group.
     */
    @Child(name = "unitPrice", type = { Money.class }, order = 14, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Fee, charge or cost per item", formalDefinition = "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.")
    protected Money unitPrice;

    /**
     * A real number that represents a multiplier used in determining the overall
     * value of services delivered and/or goods received. The concept of a Factor
     * allows for a discount or surcharge multiplier to be applied to a monetary
     * amount.
     */
    @Child(name = "factor", type = {
        DecimalType.class }, order = 15, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Price scaling factor", formalDefinition = "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.")
    protected DecimalType factor;

    /**
     * The quantity times the unit price for an additional service or product or
     * charge.
     */
    @Child(name = "net", type = { Money.class }, order = 16, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Total item cost", formalDefinition = "The quantity times the unit price for an additional service or product or charge.")
    protected Money net;

    /**
     * Unique Device Identifiers associated with this line item.
     */
    @Child(name = "udi", type = {
        Device.class }, order = 17, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Unique device identifier", formalDefinition = "Unique Device Identifiers associated with this line item.")
    protected List<Reference> udi;

    /**
     * Physical service site on the patient (limb, tooth, etc.).
     */
    @Child(name = "bodySite", type = {
        CodeableConcept.class }, order = 18, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Anatomical location", formalDefinition = "Physical service site on the patient (limb, tooth, etc.).")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/tooth")
    protected CodeableConcept bodySite;

    /**
     * A region or surface of the bodySite, e.g. limb region or tooth surface(s).
     */
    @Child(name = "subSite", type = {
        CodeableConcept.class }, order = 19, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Anatomical sub-location", formalDefinition = "A region or surface of the bodySite, e.g. limb region or tooth surface(s).")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/surface")
    protected List<CodeableConcept> subSite;

    /**
     * A billed item may include goods or services provided in multiple encounters.
     */
    @Child(name = "encounter", type = {
        Encounter.class }, order = 20, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Encounters related to this billed item", formalDefinition = "A billed item may include goods or services provided in multiple encounters.")
    protected List<Reference> encounter;

    /**
     * The numbers associated with notes below which apply to the adjudication of
     * this item.
     */
    @Child(name = "noteNumber", type = {
        PositiveIntType.class }, order = 21, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable note numbers", formalDefinition = "The numbers associated with notes below which apply to the adjudication of this item.")
    protected List<PositiveIntType> noteNumber;

    /**
     * If this item is a group then the values here are a summary of the
     * adjudication of the detail items. If this item is a simple product or service
     * then this is the result of the adjudication of this item.
     */
    @Child(name = "adjudication", type = {}, order = 22, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Adjudication details", formalDefinition = "If this item is a group then the values here are a summary of the adjudication of the detail items. If this item is a simple product or service then this is the result of the adjudication of this item.")
    protected List<AdjudicationComponent> adjudication;

    /**
     * Second-tier of goods and services.
     */
    @Child(name = "detail", type = {}, order = 23, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Additional items", formalDefinition = "Second-tier of goods and services.")
    protected List<DetailComponent> detail;

    private static final long serialVersionUID = -1350957145L;

    /**
     * Constructor
     */
    public ItemComponent() {
      super();
    }

    /**
     * Constructor
     */
    public ItemComponent(int sequence, CodeableConcept productOrService) {
      super();
      this.setSequence(sequence);
      this.setProductOrService(productOrService);
    }

    /**
     * @return {@link #sequence} (A number to uniquely identify item entries.). This
     *         is the underlying object with id, value and extensions. The accessor
     *         "getSequence" gives direct access to the value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A number to uniquely identify item entries.).
     *              This is the underlying object with id, value and extensions. The
     *              accessor "getSequence" gives direct access to the value
     */
    public ItemComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A number to uniquely identify item entries.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A number to uniquely identify item entries.
     */
    public ItemComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #careTeamSequence} (Care team members related to this service
     *         or product.)
     */
    public List<PositiveIntType> getCareTeamSequence() {
      if (this.careTeamSequence == null)
        this.careTeamSequence = new ArrayList<PositiveIntType>();
      return this.careTeamSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setCareTeamSequence(List<PositiveIntType> theCareTeamSequence) {
      this.careTeamSequence = theCareTeamSequence;
      return this;
    }

    public boolean hasCareTeamSequence() {
      if (this.careTeamSequence == null)
        return false;
      for (PositiveIntType item : this.careTeamSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #careTeamSequence} (Care team members related to this service
     *         or product.)
     */
    public PositiveIntType addCareTeamSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.careTeamSequence == null)
        this.careTeamSequence = new ArrayList<PositiveIntType>();
      this.careTeamSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #careTeamSequence} (Care team members related to this
     *              service or product.)
     */
    public ItemComponent addCareTeamSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.careTeamSequence == null)
        this.careTeamSequence = new ArrayList<PositiveIntType>();
      this.careTeamSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #careTeamSequence} (Care team members related to this
     *              service or product.)
     */
    public boolean hasCareTeamSequence(int value) {
      if (this.careTeamSequence == null)
        return false;
      for (PositiveIntType v : this.careTeamSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #diagnosisSequence} (Diagnoses applicable for this service or
     *         product.)
     */
    public List<PositiveIntType> getDiagnosisSequence() {
      if (this.diagnosisSequence == null)
        this.diagnosisSequence = new ArrayList<PositiveIntType>();
      return this.diagnosisSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setDiagnosisSequence(List<PositiveIntType> theDiagnosisSequence) {
      this.diagnosisSequence = theDiagnosisSequence;
      return this;
    }

    public boolean hasDiagnosisSequence() {
      if (this.diagnosisSequence == null)
        return false;
      for (PositiveIntType item : this.diagnosisSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #diagnosisSequence} (Diagnoses applicable for this service or
     *         product.)
     */
    public PositiveIntType addDiagnosisSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.diagnosisSequence == null)
        this.diagnosisSequence = new ArrayList<PositiveIntType>();
      this.diagnosisSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #diagnosisSequence} (Diagnoses applicable for this
     *              service or product.)
     */
    public ItemComponent addDiagnosisSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.diagnosisSequence == null)
        this.diagnosisSequence = new ArrayList<PositiveIntType>();
      this.diagnosisSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #diagnosisSequence} (Diagnoses applicable for this
     *              service or product.)
     */
    public boolean hasDiagnosisSequence(int value) {
      if (this.diagnosisSequence == null)
        return false;
      for (PositiveIntType v : this.diagnosisSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #procedureSequence} (Procedures applicable for this service or
     *         product.)
     */
    public List<PositiveIntType> getProcedureSequence() {
      if (this.procedureSequence == null)
        this.procedureSequence = new ArrayList<PositiveIntType>();
      return this.procedureSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setProcedureSequence(List<PositiveIntType> theProcedureSequence) {
      this.procedureSequence = theProcedureSequence;
      return this;
    }

    public boolean hasProcedureSequence() {
      if (this.procedureSequence == null)
        return false;
      for (PositiveIntType item : this.procedureSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #procedureSequence} (Procedures applicable for this service or
     *         product.)
     */
    public PositiveIntType addProcedureSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.procedureSequence == null)
        this.procedureSequence = new ArrayList<PositiveIntType>();
      this.procedureSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #procedureSequence} (Procedures applicable for this
     *              service or product.)
     */
    public ItemComponent addProcedureSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.procedureSequence == null)
        this.procedureSequence = new ArrayList<PositiveIntType>();
      this.procedureSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #procedureSequence} (Procedures applicable for this
     *              service or product.)
     */
    public boolean hasProcedureSequence(int value) {
      if (this.procedureSequence == null)
        return false;
      for (PositiveIntType v : this.procedureSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #informationSequence} (Exceptions, special conditions and
     *         supporting information applicable for this service or product.)
     */
    public List<PositiveIntType> getInformationSequence() {
      if (this.informationSequence == null)
        this.informationSequence = new ArrayList<PositiveIntType>();
      return this.informationSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setInformationSequence(List<PositiveIntType> theInformationSequence) {
      this.informationSequence = theInformationSequence;
      return this;
    }

    public boolean hasInformationSequence() {
      if (this.informationSequence == null)
        return false;
      for (PositiveIntType item : this.informationSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #informationSequence} (Exceptions, special conditions and
     *         supporting information applicable for this service or product.)
     */
    public PositiveIntType addInformationSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.informationSequence == null)
        this.informationSequence = new ArrayList<PositiveIntType>();
      this.informationSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #informationSequence} (Exceptions, special conditions and
     *              supporting information applicable for this service or product.)
     */
    public ItemComponent addInformationSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.informationSequence == null)
        this.informationSequence = new ArrayList<PositiveIntType>();
      this.informationSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #informationSequence} (Exceptions, special conditions and
     *              supporting information applicable for this service or product.)
     */
    public boolean hasInformationSequence(int value) {
      if (this.informationSequence == null)
        return false;
      for (PositiveIntType v : this.informationSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #revenue} (The type of revenue or cost center providing the
     *         product and/or service.)
     */
    public CodeableConcept getRevenue() {
      if (this.revenue == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.revenue");
        else if (Configuration.doAutoCreate())
          this.revenue = new CodeableConcept(); // cc
      return this.revenue;
    }

    public boolean hasRevenue() {
      return this.revenue != null && !this.revenue.isEmpty();
    }

    /**
     * @param value {@link #revenue} (The type of revenue or cost center providing
     *              the product and/or service.)
     */
    public ItemComponent setRevenue(CodeableConcept value) {
      this.revenue = value;
      return this;
    }

    /**
     * @return {@link #category} (Code to identify the general type of benefits
     *         under which products and services are provided.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (Code to identify the general type of benefits
     *              under which products and services are provided.)
     */
    public ItemComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #productOrService} (When the value is a group code then this
     *         item collects a set of related claim details, otherwise this contains
     *         the product, service, drug or other billing code for the item.)
     */
    public CodeableConcept getProductOrService() {
      if (this.productOrService == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.productOrService");
        else if (Configuration.doAutoCreate())
          this.productOrService = new CodeableConcept(); // cc
      return this.productOrService;
    }

    public boolean hasProductOrService() {
      return this.productOrService != null && !this.productOrService.isEmpty();
    }

    /**
     * @param value {@link #productOrService} (When the value is a group code then
     *              this item collects a set of related claim details, otherwise
     *              this contains the product, service, drug or other billing code
     *              for the item.)
     */
    public ItemComponent setProductOrService(CodeableConcept value) {
      this.productOrService = value;
      return this;
    }

    /**
     * @return {@link #modifier} (Item typification or modifiers codes to convey
     *         additional context for the product or service.)
     */
    public List<CodeableConcept> getModifier() {
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      return this.modifier;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setModifier(List<CodeableConcept> theModifier) {
      this.modifier = theModifier;
      return this;
    }

    public boolean hasModifier() {
      if (this.modifier == null)
        return false;
      for (CodeableConcept item : this.modifier)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addModifier() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return t;
    }

    public ItemComponent addModifier(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #modifier}, creating
     *         it if it does not already exist {3}
     */
    public CodeableConcept getModifierFirstRep() {
      if (getModifier().isEmpty()) {
        addModifier();
      }
      return getModifier().get(0);
    }

    /**
     * @return {@link #programCode} (Identifies the program under which this may be
     *         recovered.)
     */
    public List<CodeableConcept> getProgramCode() {
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      return this.programCode;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setProgramCode(List<CodeableConcept> theProgramCode) {
      this.programCode = theProgramCode;
      return this;
    }

    public boolean hasProgramCode() {
      if (this.programCode == null)
        return false;
      for (CodeableConcept item : this.programCode)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addProgramCode() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return t;
    }

    public ItemComponent addProgramCode(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #programCode},
     *         creating it if it does not already exist {3}
     */
    public CodeableConcept getProgramCodeFirstRep() {
      if (getProgramCode().isEmpty()) {
        addProgramCode();
      }
      return getProgramCode().get(0);
    }

    /**
     * @return {@link #serviced} (The date or dates when the service or product was
     *         supplied, performed or completed.)
     */
    public DataType getServiced() {
      return this.serviced;
    }

    /**
     * @return {@link #serviced} (The date or dates when the service or product was
     *         supplied, performed or completed.)
     */
    public DateType getServicedDateType() throws FHIRException {
      if (this.serviced == null)
        this.serviced = new DateType();
      if (!(this.serviced instanceof DateType))
        throw new FHIRException("Type mismatch: the type DateType was expected, but "
            + this.serviced.getClass().getName() + " was encountered");
      return (DateType) this.serviced;
    }

    public boolean hasServicedDateType() {
        return this.serviced instanceof DateType;
    }

    /**
     * @return {@link #serviced} (The date or dates when the service or product was
     *         supplied, performed or completed.)
     */
    public Period getServicedPeriod() throws FHIRException {
      if (this.serviced == null)
        this.serviced = new Period();
      if (!(this.serviced instanceof Period))
        throw new FHIRException("Type mismatch: the type Period was expected, but " + this.serviced.getClass().getName()
            + " was encountered");
      return (Period) this.serviced;
    }

    public boolean hasServicedPeriod() {
        return this.serviced instanceof Period;
    }

    public boolean hasServiced() {
      return this.serviced != null && !this.serviced.isEmpty();
    }

    /**
     * @param value {@link #serviced} (The date or dates when the service or product
     *              was supplied, performed or completed.)
     */
    public ItemComponent setServiced(DataType value) {
      if (value != null && !(value instanceof DateType || value instanceof Period))
        throw new Error("Not the right type for ExplanationOfBenefit.item.serviced[x]: " + value.fhirType());
      this.serviced = value;
      return this;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public DataType getLocation() {
      return this.location;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public CodeableConcept getLocationCodeableConcept() throws FHIRException {
      if (this.location == null)
        this.location = new CodeableConcept();
      if (!(this.location instanceof CodeableConcept))
        throw new FHIRException("Type mismatch: the type CodeableConcept was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (CodeableConcept) this.location;
    }

    public boolean hasLocationCodeableConcept() {
        return this.location instanceof CodeableConcept;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public Address getLocationAddress() throws FHIRException {
      if (this.location == null)
        this.location = new Address();
      if (!(this.location instanceof Address))
        throw new FHIRException("Type mismatch: the type Address was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (Address) this.location;
    }

    public boolean hasLocationAddress() {
        return this.location instanceof Address;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public Reference getLocationReference() throws FHIRException {
      if (this.location == null)
        this.location = new Reference();
      if (!(this.location instanceof Reference))
        throw new FHIRException("Type mismatch: the type Reference was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (Reference) this.location;
    }

    public boolean hasLocationReference() {
        return this.location instanceof Reference;
    }

    public boolean hasLocation() {
      return this.location != null && !this.location.isEmpty();
    }

    /**
     * @param value {@link #location} (Where the product or service was provided.)
     */
    public ItemComponent setLocation(DataType value) {
      if (value != null
          && !(value instanceof CodeableConcept || value instanceof Address || value instanceof Reference))
        throw new Error("Not the right type for ExplanationOfBenefit.item.location[x]: " + value.fhirType());
      this.location = value;
      return this;
    }

    /**
     * @return {@link #quantity} (The number of repetitions of a service or
     *         product.)
     */
    public Quantity getQuantity() {
      if (this.quantity == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.quantity");
        else if (Configuration.doAutoCreate())
          this.quantity = new Quantity(); // cc
      return this.quantity;
    }

    public boolean hasQuantity() {
      return this.quantity != null && !this.quantity.isEmpty();
    }

    /**
     * @param value {@link #quantity} (The number of repetitions of a service or
     *              product.)
     */
    public ItemComponent setQuantity(Quantity value) {
      this.quantity = value;
      return this;
    }

    /**
     * @return {@link #unitPrice} (If the item is not a group then this is the fee
     *         for the product or service, otherwise this is the total of the fees
     *         for the details of the group.)
     */
    public Money getUnitPrice() {
      if (this.unitPrice == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.unitPrice");
        else if (Configuration.doAutoCreate())
          this.unitPrice = new Money(); // cc
      return this.unitPrice;
    }

    public boolean hasUnitPrice() {
      return this.unitPrice != null && !this.unitPrice.isEmpty();
    }

    /**
     * @param value {@link #unitPrice} (If the item is not a group then this is the
     *              fee for the product or service, otherwise this is the total of
     *              the fees for the details of the group.)
     */
    public ItemComponent setUnitPrice(Money value) {
      this.unitPrice = value;
      return this;
    }

    /**
     * @return {@link #factor} (A real number that represents a multiplier used in
     *         determining the overall value of services delivered and/or goods
     *         received. The concept of a Factor allows for a discount or surcharge
     *         multiplier to be applied to a monetary amount.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFactor" gives direct access to the value
     */
    public DecimalType getFactorElement() {
      if (this.factor == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.factor");
        else if (Configuration.doAutoCreate())
          this.factor = new DecimalType(); // bb
      return this.factor;
    }

    public boolean hasFactorElement() {
      return this.factor != null && !this.factor.isEmpty();
    }

    public boolean hasFactor() {
      return this.factor != null && !this.factor.isEmpty();
    }

    /**
     * @param value {@link #factor} (A real number that represents a multiplier used
     *              in determining the overall value of services delivered and/or
     *              goods received. The concept of a Factor allows for a discount or
     *              surcharge multiplier to be applied to a monetary amount.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getFactor" gives direct access to the value
     */
    public ItemComponent setFactorElement(DecimalType value) {
      this.factor = value;
      return this;
    }

    /**
     * @return A real number that represents a multiplier used in determining the
     *         overall value of services delivered and/or goods received. The
     *         concept of a Factor allows for a discount or surcharge multiplier to
     *         be applied to a monetary amount.
     */
    public BigDecimal getFactor() {
      return this.factor == null ? null : this.factor.getValue();
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public ItemComponent setFactor(BigDecimal value) {
      if (value == null)
        this.factor = null;
      else {
        if (this.factor == null)
          this.factor = new DecimalType();
        this.factor.setValue(value);
      }
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public ItemComponent setFactor(long value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public ItemComponent setFactor(double value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @return {@link #net} (The quantity times the unit price for an additional
     *         service or product or charge.)
     */
    public Money getNet() {
      if (this.net == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.net");
        else if (Configuration.doAutoCreate())
          this.net = new Money(); // cc
      return this.net;
    }

    public boolean hasNet() {
      return this.net != null && !this.net.isEmpty();
    }

    /**
     * @param value {@link #net} (The quantity times the unit price for an
     *              additional service or product or charge.)
     */
    public ItemComponent setNet(Money value) {
      this.net = value;
      return this;
    }

    /**
     * @return {@link #udi} (Unique Device Identifiers associated with this line
     *         item.)
     */
    public List<Reference> getUdi() {
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      return this.udi;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setUdi(List<Reference> theUdi) {
      this.udi = theUdi;
      return this;
    }

    public boolean hasUdi() {
      if (this.udi == null)
        return false;
      for (Reference item : this.udi)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public Reference addUdi() { // 3
      Reference t = new Reference();
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return t;
    }

    public ItemComponent addUdi(Reference t) { // 3
      if (t == null)
        return this;
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #udi}, creating it if
     *         it does not already exist {3}
     */
    public Reference getUdiFirstRep() {
      if (getUdi().isEmpty()) {
        addUdi();
      }
      return getUdi().get(0);
    }

    /**
     * @return {@link #bodySite} (Physical service site on the patient (limb, tooth,
     *         etc.).)
     */
    public CodeableConcept getBodySite() {
      if (this.bodySite == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create ItemComponent.bodySite");
        else if (Configuration.doAutoCreate())
          this.bodySite = new CodeableConcept(); // cc
      return this.bodySite;
    }

    public boolean hasBodySite() {
      return this.bodySite != null && !this.bodySite.isEmpty();
    }

    /**
     * @param value {@link #bodySite} (Physical service site on the patient (limb,
     *              tooth, etc.).)
     */
    public ItemComponent setBodySite(CodeableConcept value) {
      this.bodySite = value;
      return this;
    }

    /**
     * @return {@link #subSite} (A region or surface of the bodySite, e.g. limb
     *         region or tooth surface(s).)
     */
    public List<CodeableConcept> getSubSite() {
      if (this.subSite == null)
        this.subSite = new ArrayList<CodeableConcept>();
      return this.subSite;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setSubSite(List<CodeableConcept> theSubSite) {
      this.subSite = theSubSite;
      return this;
    }

    public boolean hasSubSite() {
      if (this.subSite == null)
        return false;
      for (CodeableConcept item : this.subSite)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addSubSite() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.subSite == null)
        this.subSite = new ArrayList<CodeableConcept>();
      this.subSite.add(t);
      return t;
    }

    public ItemComponent addSubSite(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.subSite == null)
        this.subSite = new ArrayList<CodeableConcept>();
      this.subSite.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #subSite}, creating it
     *         if it does not already exist {3}
     */
    public CodeableConcept getSubSiteFirstRep() {
      if (getSubSite().isEmpty()) {
        addSubSite();
      }
      return getSubSite().get(0);
    }

    /**
     * @return {@link #encounter} (A billed item may include goods or services
     *         provided in multiple encounters.)
     */
    public List<Reference> getEncounter() {
      if (this.encounter == null)
        this.encounter = new ArrayList<Reference>();
      return this.encounter;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setEncounter(List<Reference> theEncounter) {
      this.encounter = theEncounter;
      return this;
    }

    public boolean hasEncounter() {
      if (this.encounter == null)
        return false;
      for (Reference item : this.encounter)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public Reference addEncounter() { // 3
      Reference t = new Reference();
      if (this.encounter == null)
        this.encounter = new ArrayList<Reference>();
      this.encounter.add(t);
      return t;
    }

    public ItemComponent addEncounter(Reference t) { // 3
      if (t == null)
        return this;
      if (this.encounter == null)
        this.encounter = new ArrayList<Reference>();
      this.encounter.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #encounter}, creating
     *         it if it does not already exist {3}
     */
    public Reference getEncounterFirstRep() {
      if (getEncounter().isEmpty()) {
        addEncounter();
      }
      return getEncounter().get(0);
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public List<PositiveIntType> getNoteNumber() {
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      return this.noteNumber;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setNoteNumber(List<PositiveIntType> theNoteNumber) {
      this.noteNumber = theNoteNumber;
      return this;
    }

    public boolean hasNoteNumber() {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType item : this.noteNumber)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public PositiveIntType addNoteNumberElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return t;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public ItemComponent addNoteNumber(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return this;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public boolean hasNoteNumber(int value) {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType v : this.noteNumber)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #adjudication} (If this item is a group then the values here
     *         are a summary of the adjudication of the detail items. If this item
     *         is a simple product or service then this is the result of the
     *         adjudication of this item.)
     */
    public List<AdjudicationComponent> getAdjudication() {
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      return this.adjudication;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setAdjudication(List<AdjudicationComponent> theAdjudication) {
      this.adjudication = theAdjudication;
      return this;
    }

    public boolean hasAdjudication() {
      if (this.adjudication == null)
        return false;
      for (AdjudicationComponent item : this.adjudication)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AdjudicationComponent addAdjudication() { // 3
      AdjudicationComponent t = new AdjudicationComponent();
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return t;
    }

    public ItemComponent addAdjudication(AdjudicationComponent t) { // 3
      if (t == null)
        return this;
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #adjudication},
     *         creating it if it does not already exist {3}
     */
    public AdjudicationComponent getAdjudicationFirstRep() {
      if (getAdjudication().isEmpty()) {
        addAdjudication();
      }
      return getAdjudication().get(0);
    }

    /**
     * @return {@link #detail} (Second-tier of goods and services.)
     */
    public List<DetailComponent> getDetail() {
      if (this.detail == null)
        this.detail = new ArrayList<DetailComponent>();
      return this.detail;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public ItemComponent setDetail(List<DetailComponent> theDetail) {
      this.detail = theDetail;
      return this;
    }

    public boolean hasDetail() {
      if (this.detail == null)
        return false;
      for (DetailComponent item : this.detail)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public DetailComponent addDetail() { // 3
      DetailComponent t = new DetailComponent();
      if (this.detail == null)
        this.detail = new ArrayList<DetailComponent>();
      this.detail.add(t);
      return t;
    }

    public ItemComponent addDetail(DetailComponent t) { // 3
      if (t == null)
        return this;
      if (this.detail == null)
        this.detail = new ArrayList<DetailComponent>();
      this.detail.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #detail}, creating it
     *         if it does not already exist {3}
     */
    public DetailComponent getDetailFirstRep() {
      if (getDetail().isEmpty()) {
        addDetail();
      }
      return getDetail().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children
          .add(new Property("sequence", "positiveInt", "A number to uniquely identify item entries.", 0, 1, sequence));
      children.add(new Property("careTeamSequence", "positiveInt",
          "Care team members related to this service or product.", 0, java.lang.Integer.MAX_VALUE, careTeamSequence));
      children.add(new Property("diagnosisSequence", "positiveInt", "Diagnoses applicable for this service or product.",
          0, java.lang.Integer.MAX_VALUE, diagnosisSequence));
      children.add(new Property("procedureSequence", "positiveInt",
          "Procedures applicable for this service or product.", 0, java.lang.Integer.MAX_VALUE, procedureSequence));
      children.add(new Property("informationSequence", "positiveInt",
          "Exceptions, special conditions and supporting information applicable for this service or product.", 0,
          java.lang.Integer.MAX_VALUE, informationSequence));
      children.add(new Property("revenue", "CodeableConcept",
          "The type of revenue or cost center providing the product and/or service.", 0, 1, revenue));
      children.add(new Property("category", "CodeableConcept",
          "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
          category));
      children.add(new Property("productOrService", "CodeableConcept",
          "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
          0, 1, productOrService));
      children.add(new Property("modifier", "CodeableConcept",
          "Item typification or modifiers codes to convey additional context for the product or service.", 0,
          java.lang.Integer.MAX_VALUE, modifier));
      children.add(new Property("programCode", "CodeableConcept",
          "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode));
      children.add(new Property("serviced[x]", "date|Period",
          "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced));
      children.add(new Property("location[x]", "CodeableConcept|Address|Reference(Location)",
          "Where the product or service was provided.", 0, 1, location));
      children.add(
          new Property("quantity", "Quantity", "The number of repetitions of a service or product.", 0, 1, quantity));
      children.add(new Property("unitPrice", "Money",
          "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
          0, 1, unitPrice));
      children.add(new Property("factor", "decimal",
          "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
          0, 1, factor));
      children.add(new Property("net", "Money",
          "The quantity times the unit price for an additional service or product or charge.", 0, 1, net));
      children.add(new Property("udi", "Reference(Device)", "Unique Device Identifiers associated with this line item.",
          0, java.lang.Integer.MAX_VALUE, udi));
      children.add(new Property("bodySite", "CodeableConcept",
          "Physical service site on the patient (limb, tooth, etc.).", 0, 1, bodySite));
      children.add(new Property("subSite", "CodeableConcept",
          "A region or surface of the bodySite, e.g. limb region or tooth surface(s).", 0, java.lang.Integer.MAX_VALUE,
          subSite));
      children.add(new Property("encounter", "Reference(Encounter)",
          "A billed item may include goods or services provided in multiple encounters.", 0,
          java.lang.Integer.MAX_VALUE, encounter));
      children.add(new Property("noteNumber", "positiveInt",
          "The numbers associated with notes below which apply to the adjudication of this item.", 0,
          java.lang.Integer.MAX_VALUE, noteNumber));
      children.add(new Property("adjudication", "",
          "If this item is a group then the values here are a summary of the adjudication of the detail items. If this item is a simple product or service then this is the result of the adjudication of this item.",
          0, java.lang.Integer.MAX_VALUE, adjudication));
      children.add(
          new Property("detail", "", "Second-tier of goods and services.", 0, java.lang.Integer.MAX_VALUE, detail));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt", "A number to uniquely identify item entries.", 0,
            1, sequence);
      case 1070083823:
        /* careTeamSequence */ return new Property("careTeamSequence", "positiveInt",
            "Care team members related to this service or product.", 0, java.lang.Integer.MAX_VALUE, careTeamSequence);
      case -909769262:
        /* diagnosisSequence */ return new Property("diagnosisSequence", "positiveInt",
            "Diagnoses applicable for this service or product.", 0, java.lang.Integer.MAX_VALUE, diagnosisSequence);
      case -808920140:
        /* procedureSequence */ return new Property("procedureSequence", "positiveInt",
            "Procedures applicable for this service or product.", 0, java.lang.Integer.MAX_VALUE, procedureSequence);
      case -702585587:
        /* informationSequence */ return new Property("informationSequence", "positiveInt",
            "Exceptions, special conditions and supporting information applicable for this service or product.", 0,
            java.lang.Integer.MAX_VALUE, informationSequence);
      case 1099842588:
        /* revenue */ return new Property("revenue", "CodeableConcept",
            "The type of revenue or cost center providing the product and/or service.", 0, 1, revenue);
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
            category);
      case 1957227299:
        /* productOrService */ return new Property("productOrService", "CodeableConcept",
            "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
            0, 1, productOrService);
      case -615513385:
        /* modifier */ return new Property("modifier", "CodeableConcept",
            "Item typification or modifiers codes to convey additional context for the product or service.", 0,
            java.lang.Integer.MAX_VALUE, modifier);
      case 1010065041:
        /* programCode */ return new Property("programCode", "CodeableConcept",
            "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode);
      case -1927922223:
        /* serviced[x] */ return new Property("serviced[x]", "date|Period",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 1379209295:
        /* serviced */ return new Property("serviced[x]", "date|Period",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 363246749:
        /* servicedDate */ return new Property("serviced[x]", "date",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 1534966512:
        /* servicedPeriod */ return new Property("serviced[x]", "Period",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 552316075:
        /* location[x] */ return new Property("location[x]", "CodeableConcept|Address|Reference(Location)",
            "Where the product or service was provided.", 0, 1, location);
      case 1901043637:
        /* location */ return new Property("location[x]", "CodeableConcept|Address|Reference(Location)",
            "Where the product or service was provided.", 0, 1, location);
      case -1224800468:
        /* locationCodeableConcept */ return new Property("location[x]", "CodeableConcept",
            "Where the product or service was provided.", 0, 1, location);
      case -1280020865:
        /* locationAddress */ return new Property("location[x]", "Address",
            "Where the product or service was provided.", 0, 1, location);
      case 755866390:
        /* locationReference */ return new Property("location[x]", "Reference(Location)",
            "Where the product or service was provided.", 0, 1, location);
      case -1285004149:
        /* quantity */ return new Property("quantity", "Quantity", "The number of repetitions of a service or product.",
            0, 1, quantity);
      case -486196699:
        /* unitPrice */ return new Property("unitPrice", "Money",
            "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
            0, 1, unitPrice);
      case -1282148017:
        /* factor */ return new Property("factor", "decimal",
            "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
            0, 1, factor);
      case 108957:
        /* net */ return new Property("net", "Money",
            "The quantity times the unit price for an additional service or product or charge.", 0, 1, net);
      case 115642:
        /* udi */ return new Property("udi", "Reference(Device)",
            "Unique Device Identifiers associated with this line item.", 0, java.lang.Integer.MAX_VALUE, udi);
      case 1702620169:
        /* bodySite */ return new Property("bodySite", "CodeableConcept",
            "Physical service site on the patient (limb, tooth, etc.).", 0, 1, bodySite);
      case -1868566105:
        /* subSite */ return new Property("subSite", "CodeableConcept",
            "A region or surface of the bodySite, e.g. limb region or tooth surface(s).", 0,
            java.lang.Integer.MAX_VALUE, subSite);
      case 1524132147:
        /* encounter */ return new Property("encounter", "Reference(Encounter)",
            "A billed item may include goods or services provided in multiple encounters.", 0,
            java.lang.Integer.MAX_VALUE, encounter);
      case -1110033957:
        /* noteNumber */ return new Property("noteNumber", "positiveInt",
            "The numbers associated with notes below which apply to the adjudication of this item.", 0,
            java.lang.Integer.MAX_VALUE, noteNumber);
      case -231349275:
        /* adjudication */ return new Property("adjudication", "",
            "If this item is a group then the values here are a summary of the adjudication of the detail items. If this item is a simple product or service then this is the result of the adjudication of this item.",
            0, java.lang.Integer.MAX_VALUE, adjudication);
      case -1335224239:
        /* detail */ return new Property("detail", "", "Second-tier of goods and services.", 0,
            java.lang.Integer.MAX_VALUE, detail);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case 1070083823:
        /* careTeamSequence */ return this.careTeamSequence == null ? new Base[0]
            : this.careTeamSequence.toArray(new Base[this.careTeamSequence.size()]); // PositiveIntType
      case -909769262:
        /* diagnosisSequence */ return this.diagnosisSequence == null ? new Base[0]
            : this.diagnosisSequence.toArray(new Base[this.diagnosisSequence.size()]); // PositiveIntType
      case -808920140:
        /* procedureSequence */ return this.procedureSequence == null ? new Base[0]
            : this.procedureSequence.toArray(new Base[this.procedureSequence.size()]); // PositiveIntType
      case -702585587:
        /* informationSequence */ return this.informationSequence == null ? new Base[0]
            : this.informationSequence.toArray(new Base[this.informationSequence.size()]); // PositiveIntType
      case 1099842588:
        /* revenue */ return this.revenue == null ? new Base[0] : new Base[] { this.revenue }; // CodeableConcept
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case 1957227299:
        /* productOrService */ return this.productOrService == null ? new Base[0]
            : new Base[] { this.productOrService }; // CodeableConcept
      case -615513385:
        /* modifier */ return this.modifier == null ? new Base[0]
            : this.modifier.toArray(new Base[this.modifier.size()]); // CodeableConcept
      case 1010065041:
        /* programCode */ return this.programCode == null ? new Base[0]
            : this.programCode.toArray(new Base[this.programCode.size()]); // CodeableConcept
      case 1379209295:
        /* serviced */ return this.serviced == null ? new Base[0] : new Base[] { this.serviced }; // DataType
      case 1901043637:
        /* location */ return this.location == null ? new Base[0] : new Base[] { this.location }; // DataType
      case -1285004149:
        /* quantity */ return this.quantity == null ? new Base[0] : new Base[] { this.quantity }; // Quantity
      case -486196699:
        /* unitPrice */ return this.unitPrice == null ? new Base[0] : new Base[] { this.unitPrice }; // Money
      case -1282148017:
        /* factor */ return this.factor == null ? new Base[0] : new Base[] { this.factor }; // DecimalType
      case 108957:
        /* net */ return this.net == null ? new Base[0] : new Base[] { this.net }; // Money
      case 115642:
        /* udi */ return this.udi == null ? new Base[0] : this.udi.toArray(new Base[this.udi.size()]); // Reference
      case 1702620169:
        /* bodySite */ return this.bodySite == null ? new Base[0] : new Base[] { this.bodySite }; // CodeableConcept
      case -1868566105:
        /* subSite */ return this.subSite == null ? new Base[0] : this.subSite.toArray(new Base[this.subSite.size()]); // CodeableConcept
      case 1524132147:
        /* encounter */ return this.encounter == null ? new Base[0]
            : this.encounter.toArray(new Base[this.encounter.size()]); // Reference
      case -1110033957:
        /* noteNumber */ return this.noteNumber == null ? new Base[0]
            : this.noteNumber.toArray(new Base[this.noteNumber.size()]); // PositiveIntType
      case -231349275:
        /* adjudication */ return this.adjudication == null ? new Base[0]
            : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
      case -1335224239:
        /* detail */ return this.detail == null ? new Base[0] : this.detail.toArray(new Base[this.detail.size()]); // DetailComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 1070083823: // careTeamSequence
        this.getCareTeamSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -909769262: // diagnosisSequence
        this.getDiagnosisSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -808920140: // procedureSequence
        this.getProcedureSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -702585587: // informationSequence
        this.getInformationSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case 1099842588: // revenue
        this.revenue = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 1957227299: // productOrService
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -615513385: // modifier
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1010065041: // programCode
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1379209295: // serviced
        this.serviced = TypeConvertor.castToType(value); // DataType
        return value;
      case 1901043637: // location
        this.location = TypeConvertor.castToType(value); // DataType
        return value;
      case -1285004149: // quantity
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
        return value;
      case -486196699: // unitPrice
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1282148017: // factor
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      case 108957: // net
        this.net = TypeConvertor.castToMoney(value); // Money
        return value;
      case 115642: // udi
        this.getUdi().add(TypeConvertor.castToReference(value)); // Reference
        return value;
      case 1702620169: // bodySite
        this.bodySite = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -1868566105: // subSite
        this.getSubSite().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1524132147: // encounter
        this.getEncounter().add(TypeConvertor.castToReference(value)); // Reference
        return value;
      case -1110033957: // noteNumber
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -231349275: // adjudication
        this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
        return value;
      case -1335224239: // detail
        this.getDetail().add((DetailComponent) value); // DetailComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("careTeamSequence")) {
        this.getCareTeamSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("diagnosisSequence")) {
        this.getDiagnosisSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("procedureSequence")) {
        this.getProcedureSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("informationSequence")) {
        this.getInformationSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("revenue")) {
        this.revenue = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("productOrService")) {
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("modifier")) {
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("programCode")) {
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("serviced[x]")) {
        this.serviced = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("location[x]")) {
        this.location = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("quantity")) {
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
      } else if (name.equals("unitPrice")) {
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("factor")) {
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
      } else if (name.equals("net")) {
        this.net = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("udi")) {
        this.getUdi().add(TypeConvertor.castToReference(value));
      } else if (name.equals("bodySite")) {
        this.bodySite = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("subSite")) {
        this.getSubSite().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("encounter")) {
        this.getEncounter().add(TypeConvertor.castToReference(value));
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("adjudication")) {
        this.getAdjudication().add((AdjudicationComponent) value);
      } else if (name.equals("detail")) {
        this.getDetail().add((DetailComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("careTeamSequence")) {
        this.getCareTeamSequence().remove(value);
      } else if (name.equals("diagnosisSequence")) {
        this.getDiagnosisSequence().remove(value);
      } else if (name.equals("procedureSequence")) {
        this.getProcedureSequence().remove(value);
      } else if (name.equals("informationSequence")) {
        this.getInformationSequence().remove(value);
      } else if (name.equals("revenue")) {
        this.revenue = null;
      } else if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("productOrService")) {
        this.productOrService = null;
      } else if (name.equals("modifier")) {
        this.getModifier().remove(value);
      } else if (name.equals("programCode")) {
        this.getProgramCode().remove(value);
      } else if (name.equals("serviced[x]")) {
        this.serviced = null;
      } else if (name.equals("location[x]")) {
        this.location = null;
      } else if (name.equals("quantity")) {
        this.quantity = null;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = null;
      } else if (name.equals("factor")) {
        this.factor = null;
      } else if (name.equals("net")) {
        this.net = null;
      } else if (name.equals("udi")) {
        this.getUdi().remove(value);
      } else if (name.equals("bodySite")) {
        this.bodySite = null;
      } else if (name.equals("subSite")) {
        this.getSubSite().remove(value);
      } else if (name.equals("encounter")) {
        this.getEncounter().remove(value);
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().remove(value);
      } else if (name.equals("adjudication")) {
        this.getAdjudication().remove((AdjudicationComponent) value);
      } else if (name.equals("detail")) {
        this.getDetail().remove((DetailComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case 1070083823:
        return addCareTeamSequenceElement();
      case -909769262:
        return addDiagnosisSequenceElement();
      case -808920140:
        return addProcedureSequenceElement();
      case -702585587:
        return addInformationSequenceElement();
      case 1099842588:
        return getRevenue();
      case 50511102:
        return getCategory();
      case 1957227299:
        return getProductOrService();
      case -615513385:
        return addModifier();
      case 1010065041:
        return addProgramCode();
      case -1927922223:
        return getServiced();
      case 1379209295:
        return getServiced();
      case 552316075:
        return getLocation();
      case 1901043637:
        return getLocation();
      case -1285004149:
        return getQuantity();
      case -486196699:
        return getUnitPrice();
      case -1282148017:
        return getFactorElement();
      case 108957:
        return getNet();
      case 115642:
        return addUdi();
      case 1702620169:
        return getBodySite();
      case -1868566105:
        return addSubSite();
      case 1524132147:
        return addEncounter();
      case -1110033957:
        return addNoteNumberElement();
      case -231349275:
        return addAdjudication();
      case -1335224239:
        return addDetail();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case 1070083823:
        /* careTeamSequence */ return new String[] { "positiveInt" };
      case -909769262:
        /* diagnosisSequence */ return new String[] { "positiveInt" };
      case -808920140:
        /* procedureSequence */ return new String[] { "positiveInt" };
      case -702585587:
        /* informationSequence */ return new String[] { "positiveInt" };
      case 1099842588:
        /* revenue */ return new String[] { "CodeableConcept" };
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case 1957227299:
        /* productOrService */ return new String[] { "CodeableConcept" };
      case -615513385:
        /* modifier */ return new String[] { "CodeableConcept" };
      case 1010065041:
        /* programCode */ return new String[] { "CodeableConcept" };
      case 1379209295:
        /* serviced */ return new String[] { "date", "Period" };
      case 1901043637:
        /* location */ return new String[] { "CodeableConcept", "Address", "Reference" };
      case -1285004149:
        /* quantity */ return new String[] { "Quantity" };
      case -486196699:
        /* unitPrice */ return new String[] { "Money" };
      case -1282148017:
        /* factor */ return new String[] { "decimal" };
      case 108957:
        /* net */ return new String[] { "Money" };
      case 115642:
        /* udi */ return new String[] { "Reference" };
      case 1702620169:
        /* bodySite */ return new String[] { "CodeableConcept" };
      case -1868566105:
        /* subSite */ return new String[] { "CodeableConcept" };
      case 1524132147:
        /* encounter */ return new String[] { "Reference" };
      case -1110033957:
        /* noteNumber */ return new String[] { "positiveInt" };
      case -231349275:
        /* adjudication */ return new String[] {};
      case -1335224239:
        /* detail */ return new String[] {};
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.sequence");
      } else if (name.equals("careTeamSequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.careTeamSequence");
      } else if (name.equals("diagnosisSequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.diagnosisSequence");
      } else if (name.equals("procedureSequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.procedureSequence");
      } else if (name.equals("informationSequence")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.item.informationSequence");
      } else if (name.equals("revenue")) {
        this.revenue = new CodeableConcept();
        return this.revenue;
      } else if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("productOrService")) {
        this.productOrService = new CodeableConcept();
        return this.productOrService;
      } else if (name.equals("modifier")) {
        return addModifier();
      } else if (name.equals("programCode")) {
        return addProgramCode();
      } else if (name.equals("servicedDate")) {
        this.serviced = new DateType();
        return this.serviced;
      } else if (name.equals("servicedPeriod")) {
        this.serviced = new Period();
        return this.serviced;
      } else if (name.equals("locationCodeableConcept")) {
        this.location = new CodeableConcept();
        return this.location;
      } else if (name.equals("locationAddress")) {
        this.location = new Address();
        return this.location;
      } else if (name.equals("locationReference")) {
        this.location = new Reference();
        return this.location;
      } else if (name.equals("quantity")) {
        this.quantity = new Quantity();
        return this.quantity;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = new Money();
        return this.unitPrice;
      } else if (name.equals("factor")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.factor");
      } else if (name.equals("net")) {
        this.net = new Money();
        return this.net;
      } else if (name.equals("udi")) {
        return addUdi();
      } else if (name.equals("bodySite")) {
        this.bodySite = new CodeableConcept();
        return this.bodySite;
      } else if (name.equals("subSite")) {
        return addSubSite();
      } else if (name.equals("encounter")) {
        return addEncounter();
      } else if (name.equals("noteNumber")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.noteNumber");
      } else if (name.equals("adjudication")) {
        return addAdjudication();
      } else if (name.equals("detail")) {
        return addDetail();
      } else
        return super.addChild(name);
    }

    public ItemComponent copy() {
      ItemComponent dst = new ItemComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(ItemComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      if (careTeamSequence != null) {
        dst.careTeamSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : careTeamSequence)
          dst.careTeamSequence.add(i.copy());
      }
      ;
      if (diagnosisSequence != null) {
        dst.diagnosisSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : diagnosisSequence)
          dst.diagnosisSequence.add(i.copy());
      }
      ;
      if (procedureSequence != null) {
        dst.procedureSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : procedureSequence)
          dst.procedureSequence.add(i.copy());
      }
      ;
      if (informationSequence != null) {
        dst.informationSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : informationSequence)
          dst.informationSequence.add(i.copy());
      }
      ;
      dst.revenue = revenue == null ? null : revenue.copy();
      dst.category = category == null ? null : category.copy();
      dst.productOrService = productOrService == null ? null : productOrService.copy();
      if (modifier != null) {
        dst.modifier = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : modifier)
          dst.modifier.add(i.copy());
      }
      ;
      if (programCode != null) {
        dst.programCode = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : programCode)
          dst.programCode.add(i.copy());
      }
      ;
      dst.serviced = serviced == null ? null : serviced.copy();
      dst.location = location == null ? null : location.copy();
      dst.quantity = quantity == null ? null : quantity.copy();
      dst.unitPrice = unitPrice == null ? null : unitPrice.copy();
      dst.factor = factor == null ? null : factor.copy();
      dst.net = net == null ? null : net.copy();
      if (udi != null) {
        dst.udi = new ArrayList<Reference>();
        for (Reference i : udi)
          dst.udi.add(i.copy());
      }
      ;
      dst.bodySite = bodySite == null ? null : bodySite.copy();
      if (subSite != null) {
        dst.subSite = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : subSite)
          dst.subSite.add(i.copy());
      }
      ;
      if (encounter != null) {
        dst.encounter = new ArrayList<Reference>();
        for (Reference i : encounter)
          dst.encounter.add(i.copy());
      }
      ;
      if (noteNumber != null) {
        dst.noteNumber = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : noteNumber)
          dst.noteNumber.add(i.copy());
      }
      ;
      if (adjudication != null) {
        dst.adjudication = new ArrayList<AdjudicationComponent>();
        for (AdjudicationComponent i : adjudication)
          dst.adjudication.add(i.copy());
      }
      ;
      if (detail != null) {
        dst.detail = new ArrayList<DetailComponent>();
        for (DetailComponent i : detail)
          dst.detail.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof ItemComponent))
        return false;
      ItemComponent o = (ItemComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(careTeamSequence, o.careTeamSequence, true)
          && compareDeep(diagnosisSequence, o.diagnosisSequence, true)
          && compareDeep(procedureSequence, o.procedureSequence, true)
          && compareDeep(informationSequence, o.informationSequence, true) && compareDeep(revenue, o.revenue, true)
          && compareDeep(category, o.category, true) && compareDeep(productOrService, o.productOrService, true)
          && compareDeep(modifier, o.modifier, true) && compareDeep(programCode, o.programCode, true)
          && compareDeep(serviced, o.serviced, true) && compareDeep(location, o.location, true)
          && compareDeep(quantity, o.quantity, true) && compareDeep(unitPrice, o.unitPrice, true)
          && compareDeep(factor, o.factor, true) && compareDeep(net, o.net, true) && compareDeep(udi, o.udi, true)
          && compareDeep(bodySite, o.bodySite, true) && compareDeep(subSite, o.subSite, true)
          && compareDeep(encounter, o.encounter, true) && compareDeep(noteNumber, o.noteNumber, true)
          && compareDeep(adjudication, o.adjudication, true) && compareDeep(detail, o.detail, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof ItemComponent))
        return false;
      ItemComponent o = (ItemComponent) other_;
      return compareValues(sequence, o.sequence, true) && compareValues(careTeamSequence, o.careTeamSequence, true)
          && compareValues(diagnosisSequence, o.diagnosisSequence, true)
          && compareValues(procedureSequence, o.procedureSequence, true)
          && compareValues(informationSequence, o.informationSequence, true) && compareValues(factor, o.factor, true)
          && compareValues(noteNumber, o.noteNumber, true);
    }

    public boolean isEmpty() {
      return super.isEmpty()
          && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, careTeamSequence, diagnosisSequence, procedureSequence,
              informationSequence, revenue, category, productOrService, modifier, programCode, serviced, location,
              quantity, unitPrice, factor, net, udi, bodySite, subSite, encounter, noteNumber, adjudication, detail);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.item";

    }

  }

  @Block()
  public static class AdjudicationComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A code to indicate the information type of this adjudication record.
     * Information types may include: the value submitted, maximum values or
     * percentages allowed or payable under the plan, amounts that the patient is
     * responsible for in-aggregate or pertaining to this item, amounts paid by
     * other coverages, and the benefit payable for this item.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Type of adjudication information", formalDefinition = "A code to indicate the information type of this adjudication record. Information types may include: the value submitted, maximum values or percentages allowed or payable under the plan, amounts that the patient is responsible for in-aggregate or pertaining to this item, amounts paid by other coverages, and the benefit payable for this item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/adjudication")
    protected CodeableConcept category;

    /**
     * A code supporting the understanding of the adjudication result and explaining
     * variance from expected amount.
     */
    @Child(name = "reason", type = {
        CodeableConcept.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Explanation of adjudication outcome", formalDefinition = "A code supporting the understanding of the adjudication result and explaining variance from expected amount.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/adjudication-reason")
    protected CodeableConcept reason;

    /**
     * Monetary amount associated with the category.
     */
    @Child(name = "amount", type = { Money.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Monetary amount", formalDefinition = "Monetary amount associated with the category.")
    protected Money amount;

    /**
     * A non-monetary value associated with the category. Mutually exclusive to the
     * amount element above.
     */
    @Child(name = "value", type = { DecimalType.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Non-monitary value", formalDefinition = "A non-monetary value associated with the category. Mutually exclusive to the amount element above.")
    protected DecimalType value;

    private static final long serialVersionUID = 1559898786L;

    /**
     * Constructor
     */
    public AdjudicationComponent() {
      super();
    }

    /**
     * Constructor
     */
    public AdjudicationComponent(CodeableConcept category) {
      super();
      this.setCategory(category);
    }

    /**
     * @return {@link #category} (A code to indicate the information type of this
     *         adjudication record. Information types may include: the value
     *         submitted, maximum values or percentages allowed or payable under the
     *         plan, amounts that the patient is responsible for in-aggregate or
     *         pertaining to this item, amounts paid by other coverages, and the
     *         benefit payable for this item.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AdjudicationComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (A code to indicate the information type of
     *              this adjudication record. Information types may include: the
     *              value submitted, maximum values or percentages allowed or
     *              payable under the plan, amounts that the patient is responsible
     *              for in-aggregate or pertaining to this item, amounts paid by
     *              other coverages, and the benefit payable for this item.)
     */
    public AdjudicationComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #reason} (A code supporting the understanding of the
     *         adjudication result and explaining variance from expected amount.)
     */
    public CodeableConcept getReason() {
      if (this.reason == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AdjudicationComponent.reason");
        else if (Configuration.doAutoCreate())
          this.reason = new CodeableConcept(); // cc
      return this.reason;
    }

    public boolean hasReason() {
      return this.reason != null && !this.reason.isEmpty();
    }

    /**
     * @param value {@link #reason} (A code supporting the understanding of the
     *              adjudication result and explaining variance from expected
     *              amount.)
     */
    public AdjudicationComponent setReason(CodeableConcept value) {
      this.reason = value;
      return this;
    }

    /**
     * @return {@link #amount} (Monetary amount associated with the category.)
     */
    public Money getAmount() {
      if (this.amount == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AdjudicationComponent.amount");
        else if (Configuration.doAutoCreate())
          this.amount = new Money(); // cc
      return this.amount;
    }

    public boolean hasAmount() {
      return this.amount != null && !this.amount.isEmpty();
    }

    /**
     * @param value {@link #amount} (Monetary amount associated with the category.)
     */
    public AdjudicationComponent setAmount(Money value) {
      this.amount = value;
      return this;
    }

    /**
     * @return {@link #value} (A non-monetary value associated with the category.
     *         Mutually exclusive to the amount element above.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getValue" gives direct access to the value
     */
    public DecimalType getValueElement() {
      if (this.value == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AdjudicationComponent.value");
        else if (Configuration.doAutoCreate())
          this.value = new DecimalType(); // bb
      return this.value;
    }

    public boolean hasValueElement() {
      return this.value != null && !this.value.isEmpty();
    }

    public boolean hasValue() {
      return this.value != null && !this.value.isEmpty();
    }

    /**
     * @param value {@link #value} (A non-monetary value associated with the
     *              category. Mutually exclusive to the amount element above.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getValue" gives direct access to the value
     */
    public AdjudicationComponent setValueElement(DecimalType value) {
      this.value = value;
      return this;
    }

    /**
     * @return A non-monetary value associated with the category. Mutually exclusive
     *         to the amount element above.
     */
    public BigDecimal getValue() {
      return this.value == null ? null : this.value.getValue();
    }

    /**
     * @param value A non-monetary value associated with the category. Mutually
     *              exclusive to the amount element above.
     */
    public AdjudicationComponent setValue(BigDecimal value) {
      if (value == null)
        this.value = null;
      else {
        if (this.value == null)
          this.value = new DecimalType();
        this.value.setValue(value);
      }
      return this;
    }

    /**
     * @param value A non-monetary value associated with the category. Mutually
     *              exclusive to the amount element above.
     */
    public AdjudicationComponent setValue(long value) {
      this.value = new DecimalType();
      this.value.setValue(value);
      return this;
    }

    /**
     * @param value A non-monetary value associated with the category. Mutually
     *              exclusive to the amount element above.
     */
    public AdjudicationComponent setValue(double value) {
      this.value = new DecimalType();
      this.value.setValue(value);
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("category", "CodeableConcept",
          "A code to indicate the information type of this adjudication record. Information types may include: the value submitted, maximum values or percentages allowed or payable under the plan, amounts that the patient is responsible for in-aggregate or pertaining to this item, amounts paid by other coverages, and the benefit payable for this item.",
          0, 1, category));
      children.add(new Property("reason", "CodeableConcept",
          "A code supporting the understanding of the adjudication result and explaining variance from expected amount.",
          0, 1, reason));
      children.add(new Property("amount", "Money", "Monetary amount associated with the category.", 0, 1, amount));
      children.add(new Property("value", "decimal",
          "A non-monetary value associated with the category. Mutually exclusive to the amount element above.", 0, 1,
          value));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "A code to indicate the information type of this adjudication record. Information types may include: the value submitted, maximum values or percentages allowed or payable under the plan, amounts that the patient is responsible for in-aggregate or pertaining to this item, amounts paid by other coverages, and the benefit payable for this item.",
            0, 1, category);
      case -934964668:
        /* reason */ return new Property("reason", "CodeableConcept",
            "A code supporting the understanding of the adjudication result and explaining variance from expected amount.",
            0, 1, reason);
      case -1413853096:
        /* amount */ return new Property("amount", "Money", "Monetary amount associated with the category.", 0, 1,
            amount);
      case 111972721:
        /* value */ return new Property("value", "decimal",
            "A non-monetary value associated with the category. Mutually exclusive to the amount element above.", 0, 1,
            value);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case -934964668:
        /* reason */ return this.reason == null ? new Base[0] : new Base[] { this.reason }; // CodeableConcept
      case -1413853096:
        /* amount */ return this.amount == null ? new Base[0] : new Base[] { this.amount }; // Money
      case 111972721:
        /* value */ return this.value == null ? new Base[0] : new Base[] { this.value }; // DecimalType
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -934964668: // reason
        this.reason = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -1413853096: // amount
        this.amount = TypeConvertor.castToMoney(value); // Money
        return value;
      case 111972721: // value
        this.value = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("reason")) {
        this.reason = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("amount")) {
        this.amount = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("value")) {
        this.value = TypeConvertor.castToDecimal(value); // DecimalType
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("reason")) {
        this.reason = null;
      } else if (name.equals("amount")) {
        this.amount = null;
      } else if (name.equals("value")) {
        this.value = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 50511102:
        return getCategory();
      case -934964668:
        return getReason();
      case -1413853096:
        return getAmount();
      case 111972721:
        return getValueElement();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case -934964668:
        /* reason */ return new String[] { "CodeableConcept" };
      case -1413853096:
        /* amount */ return new String[] { "Money" };
      case 111972721:
        /* value */ return new String[] { "decimal" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("reason")) {
        this.reason = new CodeableConcept();
        return this.reason;
      } else if (name.equals("amount")) {
        this.amount = new Money();
        return this.amount;
      } else if (name.equals("value")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.item.adjudication.value");
      } else
        return super.addChild(name);
    }

    public AdjudicationComponent copy() {
      AdjudicationComponent dst = new AdjudicationComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(AdjudicationComponent dst) {
      super.copyValues(dst);
      dst.category = category == null ? null : category.copy();
      dst.reason = reason == null ? null : reason.copy();
      dst.amount = amount == null ? null : amount.copy();
      dst.value = value == null ? null : value.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof AdjudicationComponent))
        return false;
      AdjudicationComponent o = (AdjudicationComponent) other_;
      return compareDeep(category, o.category, true) && compareDeep(reason, o.reason, true)
          && compareDeep(amount, o.amount, true) && compareDeep(value, o.value, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof AdjudicationComponent))
        return false;
      AdjudicationComponent o = (AdjudicationComponent) other_;
      return compareValues(value, o.value, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(category, reason, amount, value);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.item.adjudication";

    }

  }

  @Block()
  public static class DetailComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A claim detail line. Either a simple (a product or service) or a 'group' of
     * sub-details which are simple items.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Product or service provided", formalDefinition = "A claim detail line. Either a simple (a product or service) or a 'group' of sub-details which are simple items.")
    protected PositiveIntType sequence;

    /**
     * The type of revenue or cost center providing the product and/or service.
     */
    @Child(name = "revenue", type = {
        CodeableConcept.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Revenue or cost center code", formalDefinition = "The type of revenue or cost center providing the product and/or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-revenue-center")
    protected CodeableConcept revenue;

    /**
     * Code to identify the general type of benefits under which products and
     * services are provided.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefit classification", formalDefinition = "Code to identify the general type of benefits under which products and services are provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-benefitcategory")
    protected CodeableConcept category;

    /**
     * When the value is a group code then this item collects a set of related claim
     * details, otherwise this contains the product, service, drug or other billing
     * code for the item.
     */
    @Child(name = "productOrService", type = {
        CodeableConcept.class }, order = 4, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Billing, service, product, or drug code", formalDefinition = "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-uscls")
    protected CodeableConcept productOrService;

    /**
     * Item typification or modifiers codes to convey additional context for the
     * product or service.
     */
    @Child(name = "modifier", type = {
        CodeableConcept.class }, order = 5, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Service/Product billing modifiers", formalDefinition = "Item typification or modifiers codes to convey additional context for the product or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-modifiers")
    protected List<CodeableConcept> modifier;

    /**
     * Identifies the program under which this may be recovered.
     */
    @Child(name = "programCode", type = {
        CodeableConcept.class }, order = 6, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Program the product or service is provided under", formalDefinition = "Identifies the program under which this may be recovered.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-program-code")
    protected List<CodeableConcept> programCode;

    /**
     * The number of repetitions of a service or product.
     */
    @Child(name = "quantity", type = { Quantity.class }, order = 7, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Count of products or services", formalDefinition = "The number of repetitions of a service or product.")
    protected Quantity quantity;

    /**
     * If the item is not a group then this is the fee for the product or service,
     * otherwise this is the total of the fees for the details of the group.
     */
    @Child(name = "unitPrice", type = { Money.class }, order = 8, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Fee, charge or cost per item", formalDefinition = "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.")
    protected Money unitPrice;

    /**
     * A real number that represents a multiplier used in determining the overall
     * value of services delivered and/or goods received. The concept of a Factor
     * allows for a discount or surcharge multiplier to be applied to a monetary
     * amount.
     */
    @Child(name = "factor", type = {
        DecimalType.class }, order = 9, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Price scaling factor", formalDefinition = "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.")
    protected DecimalType factor;

    /**
     * The quantity times the unit price for an additional service or product or
     * charge.
     */
    @Child(name = "net", type = { Money.class }, order = 10, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Total item cost", formalDefinition = "The quantity times the unit price for an additional service or product or charge.")
    protected Money net;

    /**
     * Unique Device Identifiers associated with this line item.
     */
    @Child(name = "udi", type = {
        Device.class }, order = 11, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Unique device identifier", formalDefinition = "Unique Device Identifiers associated with this line item.")
    protected List<Reference> udi;

    /**
     * The numbers associated with notes below which apply to the adjudication of
     * this item.
     */
    @Child(name = "noteNumber", type = {
        PositiveIntType.class }, order = 12, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable note numbers", formalDefinition = "The numbers associated with notes below which apply to the adjudication of this item.")
    protected List<PositiveIntType> noteNumber;

    /**
     * The adjudication results.
     */
    @Child(name = "adjudication", type = {
        AdjudicationComponent.class }, order = 13, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Detail level adjudication details", formalDefinition = "The adjudication results.")
    protected List<AdjudicationComponent> adjudication;

    /**
     * Third-tier of goods and services.
     */
    @Child(name = "subDetail", type = {}, order = 14, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Additional items", formalDefinition = "Third-tier of goods and services.")
    protected List<SubDetailComponent> subDetail;

    private static final long serialVersionUID = -318460001L;

    /**
     * Constructor
     */
    public DetailComponent() {
      super();
    }

    /**
     * Constructor
     */
    public DetailComponent(int sequence, CodeableConcept productOrService) {
      super();
      this.setSequence(sequence);
      this.setProductOrService(productOrService);
    }

    /**
     * @return {@link #sequence} (A claim detail line. Either a simple (a product or
     *         service) or a 'group' of sub-details which are simple items.). This
     *         is the underlying object with id, value and extensions. The accessor
     *         "getSequence" gives direct access to the value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A claim detail line. Either a simple (a
     *              product or service) or a 'group' of sub-details which are simple
     *              items.). This is the underlying object with id, value and
     *              extensions. The accessor "getSequence" gives direct access to
     *              the value
     */
    public DetailComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A claim detail line. Either a simple (a product or service) or a
     *         'group' of sub-details which are simple items.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A claim detail line. Either a simple (a product or service) or a
     *              'group' of sub-details which are simple items.
     */
    public DetailComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #revenue} (The type of revenue or cost center providing the
     *         product and/or service.)
     */
    public CodeableConcept getRevenue() {
      if (this.revenue == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.revenue");
        else if (Configuration.doAutoCreate())
          this.revenue = new CodeableConcept(); // cc
      return this.revenue;
    }

    public boolean hasRevenue() {
      return this.revenue != null && !this.revenue.isEmpty();
    }

    /**
     * @param value {@link #revenue} (The type of revenue or cost center providing
     *              the product and/or service.)
     */
    public DetailComponent setRevenue(CodeableConcept value) {
      this.revenue = value;
      return this;
    }

    /**
     * @return {@link #category} (Code to identify the general type of benefits
     *         under which products and services are provided.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (Code to identify the general type of benefits
     *              under which products and services are provided.)
     */
    public DetailComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #productOrService} (When the value is a group code then this
     *         item collects a set of related claim details, otherwise this contains
     *         the product, service, drug or other billing code for the item.)
     */
    public CodeableConcept getProductOrService() {
      if (this.productOrService == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.productOrService");
        else if (Configuration.doAutoCreate())
          this.productOrService = new CodeableConcept(); // cc
      return this.productOrService;
    }

    public boolean hasProductOrService() {
      return this.productOrService != null && !this.productOrService.isEmpty();
    }

    /**
     * @param value {@link #productOrService} (When the value is a group code then
     *              this item collects a set of related claim details, otherwise
     *              this contains the product, service, drug or other billing code
     *              for the item.)
     */
    public DetailComponent setProductOrService(CodeableConcept value) {
      this.productOrService = value;
      return this;
    }

    /**
     * @return {@link #modifier} (Item typification or modifiers codes to convey
     *         additional context for the product or service.)
     */
    public List<CodeableConcept> getModifier() {
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      return this.modifier;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DetailComponent setModifier(List<CodeableConcept> theModifier) {
      this.modifier = theModifier;
      return this;
    }

    public boolean hasModifier() {
      if (this.modifier == null)
        return false;
      for (CodeableConcept item : this.modifier)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addModifier() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return t;
    }

    public DetailComponent addModifier(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #modifier}, creating
     *         it if it does not already exist {3}
     */
    public CodeableConcept getModifierFirstRep() {
      if (getModifier().isEmpty()) {
        addModifier();
      }
      return getModifier().get(0);
    }

    /**
     * @return {@link #programCode} (Identifies the program under which this may be
     *         recovered.)
     */
    public List<CodeableConcept> getProgramCode() {
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      return this.programCode;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DetailComponent setProgramCode(List<CodeableConcept> theProgramCode) {
      this.programCode = theProgramCode;
      return this;
    }

    public boolean hasProgramCode() {
      if (this.programCode == null)
        return false;
      for (CodeableConcept item : this.programCode)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addProgramCode() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return t;
    }

    public DetailComponent addProgramCode(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #programCode},
     *         creating it if it does not already exist {3}
     */
    public CodeableConcept getProgramCodeFirstRep() {
      if (getProgramCode().isEmpty()) {
        addProgramCode();
      }
      return getProgramCode().get(0);
    }

    /**
     * @return {@link #quantity} (The number of repetitions of a service or
     *         product.)
     */
    public Quantity getQuantity() {
      if (this.quantity == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.quantity");
        else if (Configuration.doAutoCreate())
          this.quantity = new Quantity(); // cc
      return this.quantity;
    }

    public boolean hasQuantity() {
      return this.quantity != null && !this.quantity.isEmpty();
    }

    /**
     * @param value {@link #quantity} (The number of repetitions of a service or
     *              product.)
     */
    public DetailComponent setQuantity(Quantity value) {
      this.quantity = value;
      return this;
    }

    /**
     * @return {@link #unitPrice} (If the item is not a group then this is the fee
     *         for the product or service, otherwise this is the total of the fees
     *         for the details of the group.)
     */
    public Money getUnitPrice() {
      if (this.unitPrice == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.unitPrice");
        else if (Configuration.doAutoCreate())
          this.unitPrice = new Money(); // cc
      return this.unitPrice;
    }

    public boolean hasUnitPrice() {
      return this.unitPrice != null && !this.unitPrice.isEmpty();
    }

    /**
     * @param value {@link #unitPrice} (If the item is not a group then this is the
     *              fee for the product or service, otherwise this is the total of
     *              the fees for the details of the group.)
     */
    public DetailComponent setUnitPrice(Money value) {
      this.unitPrice = value;
      return this;
    }

    /**
     * @return {@link #factor} (A real number that represents a multiplier used in
     *         determining the overall value of services delivered and/or goods
     *         received. The concept of a Factor allows for a discount or surcharge
     *         multiplier to be applied to a monetary amount.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFactor" gives direct access to the value
     */
    public DecimalType getFactorElement() {
      if (this.factor == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.factor");
        else if (Configuration.doAutoCreate())
          this.factor = new DecimalType(); // bb
      return this.factor;
    }

    public boolean hasFactorElement() {
      return this.factor != null && !this.factor.isEmpty();
    }

    public boolean hasFactor() {
      return this.factor != null && !this.factor.isEmpty();
    }

    /**
     * @param value {@link #factor} (A real number that represents a multiplier used
     *              in determining the overall value of services delivered and/or
     *              goods received. The concept of a Factor allows for a discount or
     *              surcharge multiplier to be applied to a monetary amount.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getFactor" gives direct access to the value
     */
    public DetailComponent setFactorElement(DecimalType value) {
      this.factor = value;
      return this;
    }

    /**
     * @return A real number that represents a multiplier used in determining the
     *         overall value of services delivered and/or goods received. The
     *         concept of a Factor allows for a discount or surcharge multiplier to
     *         be applied to a monetary amount.
     */
    public BigDecimal getFactor() {
      return this.factor == null ? null : this.factor.getValue();
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public DetailComponent setFactor(BigDecimal value) {
      if (value == null)
        this.factor = null;
      else {
        if (this.factor == null)
          this.factor = new DecimalType();
        this.factor.setValue(value);
      }
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public DetailComponent setFactor(long value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public DetailComponent setFactor(double value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @return {@link #net} (The quantity times the unit price for an additional
     *         service or product or charge.)
     */
    public Money getNet() {
      if (this.net == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create DetailComponent.net");
        else if (Configuration.doAutoCreate())
          this.net = new Money(); // cc
      return this.net;
    }

    public boolean hasNet() {
      return this.net != null && !this.net.isEmpty();
    }

    /**
     * @param value {@link #net} (The quantity times the unit price for an
     *              additional service or product or charge.)
     */
    public DetailComponent setNet(Money value) {
      this.net = value;
      return this;
    }

    /**
     * @return {@link #udi} (Unique Device Identifiers associated with this line
     *         item.)
     */
    public List<Reference> getUdi() {
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      return this.udi;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DetailComponent setUdi(List<Reference> theUdi) {
      this.udi = theUdi;
      return this;
    }

    public boolean hasUdi() {
      if (this.udi == null)
        return false;
      for (Reference item : this.udi)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public Reference addUdi() { // 3
      Reference t = new Reference();
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return t;
    }

    public DetailComponent addUdi(Reference t) { // 3
      if (t == null)
        return this;
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #udi}, creating it if
     *         it does not already exist {3}
     */
    public Reference getUdiFirstRep() {
      if (getUdi().isEmpty()) {
        addUdi();
      }
      return getUdi().get(0);
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public List<PositiveIntType> getNoteNumber() {
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      return this.noteNumber;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DetailComponent setNoteNumber(List<PositiveIntType> theNoteNumber) {
      this.noteNumber = theNoteNumber;
      return this;
    }

    public boolean hasNoteNumber() {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType item : this.noteNumber)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public PositiveIntType addNoteNumberElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return t;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public DetailComponent addNoteNumber(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return this;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public boolean hasNoteNumber(int value) {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType v : this.noteNumber)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #adjudication} (The adjudication results.)
     */
    public List<AdjudicationComponent> getAdjudication() {
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      return this.adjudication;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DetailComponent setAdjudication(List<AdjudicationComponent> theAdjudication) {
      this.adjudication = theAdjudication;
      return this;
    }

    public boolean hasAdjudication() {
      if (this.adjudication == null)
        return false;
      for (AdjudicationComponent item : this.adjudication)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AdjudicationComponent addAdjudication() { // 3
      AdjudicationComponent t = new AdjudicationComponent();
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return t;
    }

    public DetailComponent addAdjudication(AdjudicationComponent t) { // 3
      if (t == null)
        return this;
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #adjudication},
     *         creating it if it does not already exist {3}
     */
    public AdjudicationComponent getAdjudicationFirstRep() {
      if (getAdjudication().isEmpty()) {
        addAdjudication();
      }
      return getAdjudication().get(0);
    }

    /**
     * @return {@link #subDetail} (Third-tier of goods and services.)
     */
    public List<SubDetailComponent> getSubDetail() {
      if (this.subDetail == null)
        this.subDetail = new ArrayList<SubDetailComponent>();
      return this.subDetail;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public DetailComponent setSubDetail(List<SubDetailComponent> theSubDetail) {
      this.subDetail = theSubDetail;
      return this;
    }

    public boolean hasSubDetail() {
      if (this.subDetail == null)
        return false;
      for (SubDetailComponent item : this.subDetail)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public SubDetailComponent addSubDetail() { // 3
      SubDetailComponent t = new SubDetailComponent();
      if (this.subDetail == null)
        this.subDetail = new ArrayList<SubDetailComponent>();
      this.subDetail.add(t);
      return t;
    }

    public DetailComponent addSubDetail(SubDetailComponent t) { // 3
      if (t == null)
        return this;
      if (this.subDetail == null)
        this.subDetail = new ArrayList<SubDetailComponent>();
      this.subDetail.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #subDetail}, creating
     *         it if it does not already exist {3}
     */
    public SubDetailComponent getSubDetailFirstRep() {
      if (getSubDetail().isEmpty()) {
        addSubDetail();
      }
      return getSubDetail().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("sequence", "positiveInt",
          "A claim detail line. Either a simple (a product or service) or a 'group' of sub-details which are simple items.",
          0, 1, sequence));
      children.add(new Property("revenue", "CodeableConcept",
          "The type of revenue or cost center providing the product and/or service.", 0, 1, revenue));
      children.add(new Property("category", "CodeableConcept",
          "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
          category));
      children.add(new Property("productOrService", "CodeableConcept",
          "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
          0, 1, productOrService));
      children.add(new Property("modifier", "CodeableConcept",
          "Item typification or modifiers codes to convey additional context for the product or service.", 0,
          java.lang.Integer.MAX_VALUE, modifier));
      children.add(new Property("programCode", "CodeableConcept",
          "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode));
      children.add(
          new Property("quantity", "Quantity", "The number of repetitions of a service or product.", 0, 1, quantity));
      children.add(new Property("unitPrice", "Money",
          "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
          0, 1, unitPrice));
      children.add(new Property("factor", "decimal",
          "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
          0, 1, factor));
      children.add(new Property("net", "Money",
          "The quantity times the unit price for an additional service or product or charge.", 0, 1, net));
      children.add(new Property("udi", "Reference(Device)", "Unique Device Identifiers associated with this line item.",
          0, java.lang.Integer.MAX_VALUE, udi));
      children.add(new Property("noteNumber", "positiveInt",
          "The numbers associated with notes below which apply to the adjudication of this item.", 0,
          java.lang.Integer.MAX_VALUE, noteNumber));
      children.add(new Property("adjudication", "@ExplanationOfBenefit.item.adjudication", "The adjudication results.",
          0, java.lang.Integer.MAX_VALUE, adjudication));
      children.add(new Property("subDetail", "", "Third-tier of goods and services.", 0, java.lang.Integer.MAX_VALUE,
          subDetail));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt",
            "A claim detail line. Either a simple (a product or service) or a 'group' of sub-details which are simple items.",
            0, 1, sequence);
      case 1099842588:
        /* revenue */ return new Property("revenue", "CodeableConcept",
            "The type of revenue or cost center providing the product and/or service.", 0, 1, revenue);
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
            category);
      case 1957227299:
        /* productOrService */ return new Property("productOrService", "CodeableConcept",
            "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
            0, 1, productOrService);
      case -615513385:
        /* modifier */ return new Property("modifier", "CodeableConcept",
            "Item typification or modifiers codes to convey additional context for the product or service.", 0,
            java.lang.Integer.MAX_VALUE, modifier);
      case 1010065041:
        /* programCode */ return new Property("programCode", "CodeableConcept",
            "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode);
      case -1285004149:
        /* quantity */ return new Property("quantity", "Quantity", "The number of repetitions of a service or product.",
            0, 1, quantity);
      case -486196699:
        /* unitPrice */ return new Property("unitPrice", "Money",
            "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
            0, 1, unitPrice);
      case -1282148017:
        /* factor */ return new Property("factor", "decimal",
            "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
            0, 1, factor);
      case 108957:
        /* net */ return new Property("net", "Money",
            "The quantity times the unit price for an additional service or product or charge.", 0, 1, net);
      case 115642:
        /* udi */ return new Property("udi", "Reference(Device)",
            "Unique Device Identifiers associated with this line item.", 0, java.lang.Integer.MAX_VALUE, udi);
      case -1110033957:
        /* noteNumber */ return new Property("noteNumber", "positiveInt",
            "The numbers associated with notes below which apply to the adjudication of this item.", 0,
            java.lang.Integer.MAX_VALUE, noteNumber);
      case -231349275:
        /* adjudication */ return new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
            "The adjudication results.", 0, java.lang.Integer.MAX_VALUE, adjudication);
      case -828829007:
        /* subDetail */ return new Property("subDetail", "", "Third-tier of goods and services.", 0,
            java.lang.Integer.MAX_VALUE, subDetail);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case 1099842588:
        /* revenue */ return this.revenue == null ? new Base[0] : new Base[] { this.revenue }; // CodeableConcept
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case 1957227299:
        /* productOrService */ return this.productOrService == null ? new Base[0]
            : new Base[] { this.productOrService }; // CodeableConcept
      case -615513385:
        /* modifier */ return this.modifier == null ? new Base[0]
            : this.modifier.toArray(new Base[this.modifier.size()]); // CodeableConcept
      case 1010065041:
        /* programCode */ return this.programCode == null ? new Base[0]
            : this.programCode.toArray(new Base[this.programCode.size()]); // CodeableConcept
      case -1285004149:
        /* quantity */ return this.quantity == null ? new Base[0] : new Base[] { this.quantity }; // Quantity
      case -486196699:
        /* unitPrice */ return this.unitPrice == null ? new Base[0] : new Base[] { this.unitPrice }; // Money
      case -1282148017:
        /* factor */ return this.factor == null ? new Base[0] : new Base[] { this.factor }; // DecimalType
      case 108957:
        /* net */ return this.net == null ? new Base[0] : new Base[] { this.net }; // Money
      case 115642:
        /* udi */ return this.udi == null ? new Base[0] : this.udi.toArray(new Base[this.udi.size()]); // Reference
      case -1110033957:
        /* noteNumber */ return this.noteNumber == null ? new Base[0]
            : this.noteNumber.toArray(new Base[this.noteNumber.size()]); // PositiveIntType
      case -231349275:
        /* adjudication */ return this.adjudication == null ? new Base[0]
            : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
      case -828829007:
        /* subDetail */ return this.subDetail == null ? new Base[0]
            : this.subDetail.toArray(new Base[this.subDetail.size()]); // SubDetailComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 1099842588: // revenue
        this.revenue = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 1957227299: // productOrService
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -615513385: // modifier
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1010065041: // programCode
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case -1285004149: // quantity
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
        return value;
      case -486196699: // unitPrice
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1282148017: // factor
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      case 108957: // net
        this.net = TypeConvertor.castToMoney(value); // Money
        return value;
      case 115642: // udi
        this.getUdi().add(TypeConvertor.castToReference(value)); // Reference
        return value;
      case -1110033957: // noteNumber
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -231349275: // adjudication
        this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
        return value;
      case -828829007: // subDetail
        this.getSubDetail().add((SubDetailComponent) value); // SubDetailComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("revenue")) {
        this.revenue = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("productOrService")) {
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("modifier")) {
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("programCode")) {
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("quantity")) {
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
      } else if (name.equals("unitPrice")) {
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("factor")) {
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
      } else if (name.equals("net")) {
        this.net = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("udi")) {
        this.getUdi().add(TypeConvertor.castToReference(value));
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("adjudication")) {
        this.getAdjudication().add((AdjudicationComponent) value);
      } else if (name.equals("subDetail")) {
        this.getSubDetail().add((SubDetailComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("revenue")) {
        this.revenue = null;
      } else if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("productOrService")) {
        this.productOrService = null;
      } else if (name.equals("modifier")) {
        this.getModifier().remove(value);
      } else if (name.equals("programCode")) {
        this.getProgramCode().remove(value);
      } else if (name.equals("quantity")) {
        this.quantity = null;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = null;
      } else if (name.equals("factor")) {
        this.factor = null;
      } else if (name.equals("net")) {
        this.net = null;
      } else if (name.equals("udi")) {
        this.getUdi().remove(value);
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().remove(value);
      } else if (name.equals("adjudication")) {
        this.getAdjudication().remove((AdjudicationComponent) value);
      } else if (name.equals("subDetail")) {
        this.getSubDetail().remove((SubDetailComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case 1099842588:
        return getRevenue();
      case 50511102:
        return getCategory();
      case 1957227299:
        return getProductOrService();
      case -615513385:
        return addModifier();
      case 1010065041:
        return addProgramCode();
      case -1285004149:
        return getQuantity();
      case -486196699:
        return getUnitPrice();
      case -1282148017:
        return getFactorElement();
      case 108957:
        return getNet();
      case 115642:
        return addUdi();
      case -1110033957:
        return addNoteNumberElement();
      case -231349275:
        return addAdjudication();
      case -828829007:
        return addSubDetail();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case 1099842588:
        /* revenue */ return new String[] { "CodeableConcept" };
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case 1957227299:
        /* productOrService */ return new String[] { "CodeableConcept" };
      case -615513385:
        /* modifier */ return new String[] { "CodeableConcept" };
      case 1010065041:
        /* programCode */ return new String[] { "CodeableConcept" };
      case -1285004149:
        /* quantity */ return new String[] { "Quantity" };
      case -486196699:
        /* unitPrice */ return new String[] { "Money" };
      case -1282148017:
        /* factor */ return new String[] { "decimal" };
      case 108957:
        /* net */ return new String[] { "Money" };
      case 115642:
        /* udi */ return new String[] { "Reference" };
      case -1110033957:
        /* noteNumber */ return new String[] { "positiveInt" };
      case -231349275:
        /* adjudication */ return new String[] { "@ExplanationOfBenefit.item.adjudication" };
      case -828829007:
        /* subDetail */ return new String[] {};
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.detail.sequence");
      } else if (name.equals("revenue")) {
        this.revenue = new CodeableConcept();
        return this.revenue;
      } else if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("productOrService")) {
        this.productOrService = new CodeableConcept();
        return this.productOrService;
      } else if (name.equals("modifier")) {
        return addModifier();
      } else if (name.equals("programCode")) {
        return addProgramCode();
      } else if (name.equals("quantity")) {
        this.quantity = new Quantity();
        return this.quantity;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = new Money();
        return this.unitPrice;
      } else if (name.equals("factor")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.detail.factor");
      } else if (name.equals("net")) {
        this.net = new Money();
        return this.net;
      } else if (name.equals("udi")) {
        return addUdi();
      } else if (name.equals("noteNumber")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.item.detail.noteNumber");
      } else if (name.equals("adjudication")) {
        return addAdjudication();
      } else if (name.equals("subDetail")) {
        return addSubDetail();
      } else
        return super.addChild(name);
    }

    public DetailComponent copy() {
      DetailComponent dst = new DetailComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(DetailComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      dst.revenue = revenue == null ? null : revenue.copy();
      dst.category = category == null ? null : category.copy();
      dst.productOrService = productOrService == null ? null : productOrService.copy();
      if (modifier != null) {
        dst.modifier = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : modifier)
          dst.modifier.add(i.copy());
      }
      ;
      if (programCode != null) {
        dst.programCode = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : programCode)
          dst.programCode.add(i.copy());
      }
      ;
      dst.quantity = quantity == null ? null : quantity.copy();
      dst.unitPrice = unitPrice == null ? null : unitPrice.copy();
      dst.factor = factor == null ? null : factor.copy();
      dst.net = net == null ? null : net.copy();
      if (udi != null) {
        dst.udi = new ArrayList<Reference>();
        for (Reference i : udi)
          dst.udi.add(i.copy());
      }
      ;
      if (noteNumber != null) {
        dst.noteNumber = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : noteNumber)
          dst.noteNumber.add(i.copy());
      }
      ;
      if (adjudication != null) {
        dst.adjudication = new ArrayList<AdjudicationComponent>();
        for (AdjudicationComponent i : adjudication)
          dst.adjudication.add(i.copy());
      }
      ;
      if (subDetail != null) {
        dst.subDetail = new ArrayList<SubDetailComponent>();
        for (SubDetailComponent i : subDetail)
          dst.subDetail.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof DetailComponent))
        return false;
      DetailComponent o = (DetailComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(revenue, o.revenue, true)
          && compareDeep(category, o.category, true) && compareDeep(productOrService, o.productOrService, true)
          && compareDeep(modifier, o.modifier, true) && compareDeep(programCode, o.programCode, true)
          && compareDeep(quantity, o.quantity, true) && compareDeep(unitPrice, o.unitPrice, true)
          && compareDeep(factor, o.factor, true) && compareDeep(net, o.net, true) && compareDeep(udi, o.udi, true)
          && compareDeep(noteNumber, o.noteNumber, true) && compareDeep(adjudication, o.adjudication, true)
          && compareDeep(subDetail, o.subDetail, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof DetailComponent))
        return false;
      DetailComponent o = (DetailComponent) other_;
      return compareValues(sequence, o.sequence, true) && compareValues(factor, o.factor, true)
          && compareValues(noteNumber, o.noteNumber, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, revenue, category, productOrService,
          modifier, programCode, quantity, unitPrice, factor, net, udi, noteNumber, adjudication, subDetail);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.item.detail";

    }

  }

  @Block()
  public static class SubDetailComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A claim detail line. Either a simple (a product or service) or a 'group' of
     * sub-details which are simple items.
     */
    @Child(name = "sequence", type = {
        PositiveIntType.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Product or service provided", formalDefinition = "A claim detail line. Either a simple (a product or service) or a 'group' of sub-details which are simple items.")
    protected PositiveIntType sequence;

    /**
     * The type of revenue or cost center providing the product and/or service.
     */
    @Child(name = "revenue", type = {
        CodeableConcept.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Revenue or cost center code", formalDefinition = "The type of revenue or cost center providing the product and/or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-revenue-center")
    protected CodeableConcept revenue;

    /**
     * Code to identify the general type of benefits under which products and
     * services are provided.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefit classification", formalDefinition = "Code to identify the general type of benefits under which products and services are provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-benefitcategory")
    protected CodeableConcept category;

    /**
     * When the value is a group code then this item collects a set of related claim
     * details, otherwise this contains the product, service, drug or other billing
     * code for the item.
     */
    @Child(name = "productOrService", type = {
        CodeableConcept.class }, order = 4, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Billing, service, product, or drug code", formalDefinition = "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-uscls")
    protected CodeableConcept productOrService;

    /**
     * Item typification or modifiers codes to convey additional context for the
     * product or service.
     */
    @Child(name = "modifier", type = {
        CodeableConcept.class }, order = 5, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Service/Product billing modifiers", formalDefinition = "Item typification or modifiers codes to convey additional context for the product or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-modifiers")
    protected List<CodeableConcept> modifier;

    /**
     * Identifies the program under which this may be recovered.
     */
    @Child(name = "programCode", type = {
        CodeableConcept.class }, order = 6, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Program the product or service is provided under", formalDefinition = "Identifies the program under which this may be recovered.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-program-code")
    protected List<CodeableConcept> programCode;

    /**
     * The number of repetitions of a service or product.
     */
    @Child(name = "quantity", type = { Quantity.class }, order = 7, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Count of products or services", formalDefinition = "The number of repetitions of a service or product.")
    protected Quantity quantity;

    /**
     * If the item is not a group then this is the fee for the product or service,
     * otherwise this is the total of the fees for the details of the group.
     */
    @Child(name = "unitPrice", type = { Money.class }, order = 8, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Fee, charge or cost per item", formalDefinition = "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.")
    protected Money unitPrice;

    /**
     * A real number that represents a multiplier used in determining the overall
     * value of services delivered and/or goods received. The concept of a Factor
     * allows for a discount or surcharge multiplier to be applied to a monetary
     * amount.
     */
    @Child(name = "factor", type = {
        DecimalType.class }, order = 9, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Price scaling factor", formalDefinition = "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.")
    protected DecimalType factor;

    /**
     * The quantity times the unit price for an additional service or product or
     * charge.
     */
    @Child(name = "net", type = { Money.class }, order = 10, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Total item cost", formalDefinition = "The quantity times the unit price for an additional service or product or charge.")
    protected Money net;

    /**
     * Unique Device Identifiers associated with this line item.
     */
    @Child(name = "udi", type = {
        Device.class }, order = 11, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Unique device identifier", formalDefinition = "Unique Device Identifiers associated with this line item.")
    protected List<Reference> udi;

    /**
     * The numbers associated with notes below which apply to the adjudication of
     * this item.
     */
    @Child(name = "noteNumber", type = {
        PositiveIntType.class }, order = 12, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable note numbers", formalDefinition = "The numbers associated with notes below which apply to the adjudication of this item.")
    protected List<PositiveIntType> noteNumber;

    /**
     * The adjudication results.
     */
    @Child(name = "adjudication", type = {
        AdjudicationComponent.class }, order = 13, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Subdetail level adjudication details", formalDefinition = "The adjudication results.")
    protected List<AdjudicationComponent> adjudication;

    private static final long serialVersionUID = -579551678L;

    /**
     * Constructor
     */
    public SubDetailComponent() {
      super();
    }

    /**
     * Constructor
     */
    public SubDetailComponent(int sequence, CodeableConcept productOrService) {
      super();
      this.setSequence(sequence);
      this.setProductOrService(productOrService);
    }

    /**
     * @return {@link #sequence} (A claim detail line. Either a simple (a product or
     *         service) or a 'group' of sub-details which are simple items.). This
     *         is the underlying object with id, value and extensions. The accessor
     *         "getSequence" gives direct access to the value
     */
    public PositiveIntType getSequenceElement() {
      if (this.sequence == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.sequence");
        else if (Configuration.doAutoCreate())
          this.sequence = new PositiveIntType(); // bb
      return this.sequence;
    }

    public boolean hasSequenceElement() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    public boolean hasSequence() {
      return this.sequence != null && !this.sequence.isEmpty();
    }

    /**
     * @param value {@link #sequence} (A claim detail line. Either a simple (a
     *              product or service) or a 'group' of sub-details which are simple
     *              items.). This is the underlying object with id, value and
     *              extensions. The accessor "getSequence" gives direct access to
     *              the value
     */
    public SubDetailComponent setSequenceElement(PositiveIntType value) {
      this.sequence = value;
      return this;
    }

    /**
     * @return A claim detail line. Either a simple (a product or service) or a
     *         'group' of sub-details which are simple items.
     */
    public int getSequence() {
      return this.sequence == null || this.sequence.isEmpty() ? 0 : this.sequence.getValue();
    }

    /**
     * @param value A claim detail line. Either a simple (a product or service) or a
     *              'group' of sub-details which are simple items.
     */
    public SubDetailComponent setSequence(int value) {
      if (this.sequence == null)
        this.sequence = new PositiveIntType();
      this.sequence.setValue(value);
      return this;
    }

    /**
     * @return {@link #revenue} (The type of revenue or cost center providing the
     *         product and/or service.)
     */
    public CodeableConcept getRevenue() {
      if (this.revenue == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.revenue");
        else if (Configuration.doAutoCreate())
          this.revenue = new CodeableConcept(); // cc
      return this.revenue;
    }

    public boolean hasRevenue() {
      return this.revenue != null && !this.revenue.isEmpty();
    }

    /**
     * @param value {@link #revenue} (The type of revenue or cost center providing
     *              the product and/or service.)
     */
    public SubDetailComponent setRevenue(CodeableConcept value) {
      this.revenue = value;
      return this;
    }

    /**
     * @return {@link #category} (Code to identify the general type of benefits
     *         under which products and services are provided.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (Code to identify the general type of benefits
     *              under which products and services are provided.)
     */
    public SubDetailComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #productOrService} (When the value is a group code then this
     *         item collects a set of related claim details, otherwise this contains
     *         the product, service, drug or other billing code for the item.)
     */
    public CodeableConcept getProductOrService() {
      if (this.productOrService == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.productOrService");
        else if (Configuration.doAutoCreate())
          this.productOrService = new CodeableConcept(); // cc
      return this.productOrService;
    }

    public boolean hasProductOrService() {
      return this.productOrService != null && !this.productOrService.isEmpty();
    }

    /**
     * @param value {@link #productOrService} (When the value is a group code then
     *              this item collects a set of related claim details, otherwise
     *              this contains the product, service, drug or other billing code
     *              for the item.)
     */
    public SubDetailComponent setProductOrService(CodeableConcept value) {
      this.productOrService = value;
      return this;
    }

    /**
     * @return {@link #modifier} (Item typification or modifiers codes to convey
     *         additional context for the product or service.)
     */
    public List<CodeableConcept> getModifier() {
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      return this.modifier;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public SubDetailComponent setModifier(List<CodeableConcept> theModifier) {
      this.modifier = theModifier;
      return this;
    }

    public boolean hasModifier() {
      if (this.modifier == null)
        return false;
      for (CodeableConcept item : this.modifier)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addModifier() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return t;
    }

    public SubDetailComponent addModifier(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #modifier}, creating
     *         it if it does not already exist {3}
     */
    public CodeableConcept getModifierFirstRep() {
      if (getModifier().isEmpty()) {
        addModifier();
      }
      return getModifier().get(0);
    }

    /**
     * @return {@link #programCode} (Identifies the program under which this may be
     *         recovered.)
     */
    public List<CodeableConcept> getProgramCode() {
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      return this.programCode;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public SubDetailComponent setProgramCode(List<CodeableConcept> theProgramCode) {
      this.programCode = theProgramCode;
      return this;
    }

    public boolean hasProgramCode() {
      if (this.programCode == null)
        return false;
      for (CodeableConcept item : this.programCode)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addProgramCode() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return t;
    }

    public SubDetailComponent addProgramCode(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #programCode},
     *         creating it if it does not already exist {3}
     */
    public CodeableConcept getProgramCodeFirstRep() {
      if (getProgramCode().isEmpty()) {
        addProgramCode();
      }
      return getProgramCode().get(0);
    }

    /**
     * @return {@link #quantity} (The number of repetitions of a service or
     *         product.)
     */
    public Quantity getQuantity() {
      if (this.quantity == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.quantity");
        else if (Configuration.doAutoCreate())
          this.quantity = new Quantity(); // cc
      return this.quantity;
    }

    public boolean hasQuantity() {
      return this.quantity != null && !this.quantity.isEmpty();
    }

    /**
     * @param value {@link #quantity} (The number of repetitions of a service or
     *              product.)
     */
    public SubDetailComponent setQuantity(Quantity value) {
      this.quantity = value;
      return this;
    }

    /**
     * @return {@link #unitPrice} (If the item is not a group then this is the fee
     *         for the product or service, otherwise this is the total of the fees
     *         for the details of the group.)
     */
    public Money getUnitPrice() {
      if (this.unitPrice == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.unitPrice");
        else if (Configuration.doAutoCreate())
          this.unitPrice = new Money(); // cc
      return this.unitPrice;
    }

    public boolean hasUnitPrice() {
      return this.unitPrice != null && !this.unitPrice.isEmpty();
    }

    /**
     * @param value {@link #unitPrice} (If the item is not a group then this is the
     *              fee for the product or service, otherwise this is the total of
     *              the fees for the details of the group.)
     */
    public SubDetailComponent setUnitPrice(Money value) {
      this.unitPrice = value;
      return this;
    }

    /**
     * @return {@link #factor} (A real number that represents a multiplier used in
     *         determining the overall value of services delivered and/or goods
     *         received. The concept of a Factor allows for a discount or surcharge
     *         multiplier to be applied to a monetary amount.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFactor" gives direct access to the value
     */
    public DecimalType getFactorElement() {
      if (this.factor == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.factor");
        else if (Configuration.doAutoCreate())
          this.factor = new DecimalType(); // bb
      return this.factor;
    }

    public boolean hasFactorElement() {
      return this.factor != null && !this.factor.isEmpty();
    }

    public boolean hasFactor() {
      return this.factor != null && !this.factor.isEmpty();
    }

    /**
     * @param value {@link #factor} (A real number that represents a multiplier used
     *              in determining the overall value of services delivered and/or
     *              goods received. The concept of a Factor allows for a discount or
     *              surcharge multiplier to be applied to a monetary amount.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getFactor" gives direct access to the value
     */
    public SubDetailComponent setFactorElement(DecimalType value) {
      this.factor = value;
      return this;
    }

    /**
     * @return A real number that represents a multiplier used in determining the
     *         overall value of services delivered and/or goods received. The
     *         concept of a Factor allows for a discount or surcharge multiplier to
     *         be applied to a monetary amount.
     */
    public BigDecimal getFactor() {
      return this.factor == null ? null : this.factor.getValue();
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public SubDetailComponent setFactor(BigDecimal value) {
      if (value == null)
        this.factor = null;
      else {
        if (this.factor == null)
          this.factor = new DecimalType();
        this.factor.setValue(value);
      }
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public SubDetailComponent setFactor(long value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public SubDetailComponent setFactor(double value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @return {@link #net} (The quantity times the unit price for an additional
     *         service or product or charge.)
     */
    public Money getNet() {
      if (this.net == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create SubDetailComponent.net");
        else if (Configuration.doAutoCreate())
          this.net = new Money(); // cc
      return this.net;
    }

    public boolean hasNet() {
      return this.net != null && !this.net.isEmpty();
    }

    /**
     * @param value {@link #net} (The quantity times the unit price for an
     *              additional service or product or charge.)
     */
    public SubDetailComponent setNet(Money value) {
      this.net = value;
      return this;
    }

    /**
     * @return {@link #udi} (Unique Device Identifiers associated with this line
     *         item.)
     */
    public List<Reference> getUdi() {
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      return this.udi;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public SubDetailComponent setUdi(List<Reference> theUdi) {
      this.udi = theUdi;
      return this;
    }

    public boolean hasUdi() {
      if (this.udi == null)
        return false;
      for (Reference item : this.udi)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public Reference addUdi() { // 3
      Reference t = new Reference();
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return t;
    }

    public SubDetailComponent addUdi(Reference t) { // 3
      if (t == null)
        return this;
      if (this.udi == null)
        this.udi = new ArrayList<Reference>();
      this.udi.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #udi}, creating it if
     *         it does not already exist {3}
     */
    public Reference getUdiFirstRep() {
      if (getUdi().isEmpty()) {
        addUdi();
      }
      return getUdi().get(0);
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public List<PositiveIntType> getNoteNumber() {
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      return this.noteNumber;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public SubDetailComponent setNoteNumber(List<PositiveIntType> theNoteNumber) {
      this.noteNumber = theNoteNumber;
      return this;
    }

    public boolean hasNoteNumber() {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType item : this.noteNumber)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public PositiveIntType addNoteNumberElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return t;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public SubDetailComponent addNoteNumber(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return this;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public boolean hasNoteNumber(int value) {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType v : this.noteNumber)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #adjudication} (The adjudication results.)
     */
    public List<AdjudicationComponent> getAdjudication() {
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      return this.adjudication;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public SubDetailComponent setAdjudication(List<AdjudicationComponent> theAdjudication) {
      this.adjudication = theAdjudication;
      return this;
    }

    public boolean hasAdjudication() {
      if (this.adjudication == null)
        return false;
      for (AdjudicationComponent item : this.adjudication)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AdjudicationComponent addAdjudication() { // 3
      AdjudicationComponent t = new AdjudicationComponent();
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return t;
    }

    public SubDetailComponent addAdjudication(AdjudicationComponent t) { // 3
      if (t == null)
        return this;
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #adjudication},
     *         creating it if it does not already exist {3}
     */
    public AdjudicationComponent getAdjudicationFirstRep() {
      if (getAdjudication().isEmpty()) {
        addAdjudication();
      }
      return getAdjudication().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("sequence", "positiveInt",
          "A claim detail line. Either a simple (a product or service) or a 'group' of sub-details which are simple items.",
          0, 1, sequence));
      children.add(new Property("revenue", "CodeableConcept",
          "The type of revenue or cost center providing the product and/or service.", 0, 1, revenue));
      children.add(new Property("category", "CodeableConcept",
          "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
          category));
      children.add(new Property("productOrService", "CodeableConcept",
          "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
          0, 1, productOrService));
      children.add(new Property("modifier", "CodeableConcept",
          "Item typification or modifiers codes to convey additional context for the product or service.", 0,
          java.lang.Integer.MAX_VALUE, modifier));
      children.add(new Property("programCode", "CodeableConcept",
          "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode));
      children.add(
          new Property("quantity", "Quantity", "The number of repetitions of a service or product.", 0, 1, quantity));
      children.add(new Property("unitPrice", "Money",
          "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
          0, 1, unitPrice));
      children.add(new Property("factor", "decimal",
          "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
          0, 1, factor));
      children.add(new Property("net", "Money",
          "The quantity times the unit price for an additional service or product or charge.", 0, 1, net));
      children.add(new Property("udi", "Reference(Device)", "Unique Device Identifiers associated with this line item.",
          0, java.lang.Integer.MAX_VALUE, udi));
      children.add(new Property("noteNumber", "positiveInt",
          "The numbers associated with notes below which apply to the adjudication of this item.", 0,
          java.lang.Integer.MAX_VALUE, noteNumber));
      children.add(new Property("adjudication", "@ExplanationOfBenefit.item.adjudication", "The adjudication results.",
          0, java.lang.Integer.MAX_VALUE, adjudication));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1349547969:
        /* sequence */ return new Property("sequence", "positiveInt",
            "A claim detail line. Either a simple (a product or service) or a 'group' of sub-details which are simple items.",
            0, 1, sequence);
      case 1099842588:
        /* revenue */ return new Property("revenue", "CodeableConcept",
            "The type of revenue or cost center providing the product and/or service.", 0, 1, revenue);
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
            category);
      case 1957227299:
        /* productOrService */ return new Property("productOrService", "CodeableConcept",
            "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
            0, 1, productOrService);
      case -615513385:
        /* modifier */ return new Property("modifier", "CodeableConcept",
            "Item typification or modifiers codes to convey additional context for the product or service.", 0,
            java.lang.Integer.MAX_VALUE, modifier);
      case 1010065041:
        /* programCode */ return new Property("programCode", "CodeableConcept",
            "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode);
      case -1285004149:
        /* quantity */ return new Property("quantity", "Quantity", "The number of repetitions of a service or product.",
            0, 1, quantity);
      case -486196699:
        /* unitPrice */ return new Property("unitPrice", "Money",
            "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
            0, 1, unitPrice);
      case -1282148017:
        /* factor */ return new Property("factor", "decimal",
            "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
            0, 1, factor);
      case 108957:
        /* net */ return new Property("net", "Money",
            "The quantity times the unit price for an additional service or product or charge.", 0, 1, net);
      case 115642:
        /* udi */ return new Property("udi", "Reference(Device)",
            "Unique Device Identifiers associated with this line item.", 0, java.lang.Integer.MAX_VALUE, udi);
      case -1110033957:
        /* noteNumber */ return new Property("noteNumber", "positiveInt",
            "The numbers associated with notes below which apply to the adjudication of this item.", 0,
            java.lang.Integer.MAX_VALUE, noteNumber);
      case -231349275:
        /* adjudication */ return new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
            "The adjudication results.", 0, java.lang.Integer.MAX_VALUE, adjudication);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return this.sequence == null ? new Base[0] : new Base[] { this.sequence }; // PositiveIntType
      case 1099842588:
        /* revenue */ return this.revenue == null ? new Base[0] : new Base[] { this.revenue }; // CodeableConcept
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case 1957227299:
        /* productOrService */ return this.productOrService == null ? new Base[0]
            : new Base[] { this.productOrService }; // CodeableConcept
      case -615513385:
        /* modifier */ return this.modifier == null ? new Base[0]
            : this.modifier.toArray(new Base[this.modifier.size()]); // CodeableConcept
      case 1010065041:
        /* programCode */ return this.programCode == null ? new Base[0]
            : this.programCode.toArray(new Base[this.programCode.size()]); // CodeableConcept
      case -1285004149:
        /* quantity */ return this.quantity == null ? new Base[0] : new Base[] { this.quantity }; // Quantity
      case -486196699:
        /* unitPrice */ return this.unitPrice == null ? new Base[0] : new Base[] { this.unitPrice }; // Money
      case -1282148017:
        /* factor */ return this.factor == null ? new Base[0] : new Base[] { this.factor }; // DecimalType
      case 108957:
        /* net */ return this.net == null ? new Base[0] : new Base[] { this.net }; // Money
      case 115642:
        /* udi */ return this.udi == null ? new Base[0] : this.udi.toArray(new Base[this.udi.size()]); // Reference
      case -1110033957:
        /* noteNumber */ return this.noteNumber == null ? new Base[0]
            : this.noteNumber.toArray(new Base[this.noteNumber.size()]); // PositiveIntType
      case -231349275:
        /* adjudication */ return this.adjudication == null ? new Base[0]
            : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1349547969: // sequence
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 1099842588: // revenue
        this.revenue = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 1957227299: // productOrService
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -615513385: // modifier
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1010065041: // programCode
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case -1285004149: // quantity
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
        return value;
      case -486196699: // unitPrice
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1282148017: // factor
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      case 108957: // net
        this.net = TypeConvertor.castToMoney(value); // Money
        return value;
      case 115642: // udi
        this.getUdi().add(TypeConvertor.castToReference(value)); // Reference
        return value;
      case -1110033957: // noteNumber
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -231349275: // adjudication
        this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("revenue")) {
        this.revenue = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("productOrService")) {
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("modifier")) {
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("programCode")) {
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("quantity")) {
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
      } else if (name.equals("unitPrice")) {
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("factor")) {
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
      } else if (name.equals("net")) {
        this.net = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("udi")) {
        this.getUdi().add(TypeConvertor.castToReference(value));
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("adjudication")) {
        this.getAdjudication().add((AdjudicationComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("sequence")) {
        this.sequence = null;
      } else if (name.equals("revenue")) {
        this.revenue = null;
      } else if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("productOrService")) {
        this.productOrService = null;
      } else if (name.equals("modifier")) {
        this.getModifier().remove(value);
      } else if (name.equals("programCode")) {
        this.getProgramCode().remove(value);
      } else if (name.equals("quantity")) {
        this.quantity = null;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = null;
      } else if (name.equals("factor")) {
        this.factor = null;
      } else if (name.equals("net")) {
        this.net = null;
      } else if (name.equals("udi")) {
        this.getUdi().remove(value);
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().remove(value);
      } else if (name.equals("adjudication")) {
        this.getAdjudication().remove((AdjudicationComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        return getSequenceElement();
      case 1099842588:
        return getRevenue();
      case 50511102:
        return getCategory();
      case 1957227299:
        return getProductOrService();
      case -615513385:
        return addModifier();
      case 1010065041:
        return addProgramCode();
      case -1285004149:
        return getQuantity();
      case -486196699:
        return getUnitPrice();
      case -1282148017:
        return getFactorElement();
      case 108957:
        return getNet();
      case 115642:
        return addUdi();
      case -1110033957:
        return addNoteNumberElement();
      case -231349275:
        return addAdjudication();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1349547969:
        /* sequence */ return new String[] { "positiveInt" };
      case 1099842588:
        /* revenue */ return new String[] { "CodeableConcept" };
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case 1957227299:
        /* productOrService */ return new String[] { "CodeableConcept" };
      case -615513385:
        /* modifier */ return new String[] { "CodeableConcept" };
      case 1010065041:
        /* programCode */ return new String[] { "CodeableConcept" };
      case -1285004149:
        /* quantity */ return new String[] { "Quantity" };
      case -486196699:
        /* unitPrice */ return new String[] { "Money" };
      case -1282148017:
        /* factor */ return new String[] { "decimal" };
      case 108957:
        /* net */ return new String[] { "Money" };
      case 115642:
        /* udi */ return new String[] { "Reference" };
      case -1110033957:
        /* noteNumber */ return new String[] { "positiveInt" };
      case -231349275:
        /* adjudication */ return new String[] { "@ExplanationOfBenefit.item.adjudication" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("sequence")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.item.detail.subDetail.sequence");
      } else if (name.equals("revenue")) {
        this.revenue = new CodeableConcept();
        return this.revenue;
      } else if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("productOrService")) {
        this.productOrService = new CodeableConcept();
        return this.productOrService;
      } else if (name.equals("modifier")) {
        return addModifier();
      } else if (name.equals("programCode")) {
        return addProgramCode();
      } else if (name.equals("quantity")) {
        this.quantity = new Quantity();
        return this.quantity;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = new Money();
        return this.unitPrice;
      } else if (name.equals("factor")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.item.detail.subDetail.factor");
      } else if (name.equals("net")) {
        this.net = new Money();
        return this.net;
      } else if (name.equals("udi")) {
        return addUdi();
      } else if (name.equals("noteNumber")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.item.detail.subDetail.noteNumber");
      } else if (name.equals("adjudication")) {
        return addAdjudication();
      } else
        return super.addChild(name);
    }

    public SubDetailComponent copy() {
      SubDetailComponent dst = new SubDetailComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(SubDetailComponent dst) {
      super.copyValues(dst);
      dst.sequence = sequence == null ? null : sequence.copy();
      dst.revenue = revenue == null ? null : revenue.copy();
      dst.category = category == null ? null : category.copy();
      dst.productOrService = productOrService == null ? null : productOrService.copy();
      if (modifier != null) {
        dst.modifier = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : modifier)
          dst.modifier.add(i.copy());
      }
      ;
      if (programCode != null) {
        dst.programCode = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : programCode)
          dst.programCode.add(i.copy());
      }
      ;
      dst.quantity = quantity == null ? null : quantity.copy();
      dst.unitPrice = unitPrice == null ? null : unitPrice.copy();
      dst.factor = factor == null ? null : factor.copy();
      dst.net = net == null ? null : net.copy();
      if (udi != null) {
        dst.udi = new ArrayList<Reference>();
        for (Reference i : udi)
          dst.udi.add(i.copy());
      }
      ;
      if (noteNumber != null) {
        dst.noteNumber = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : noteNumber)
          dst.noteNumber.add(i.copy());
      }
      ;
      if (adjudication != null) {
        dst.adjudication = new ArrayList<AdjudicationComponent>();
        for (AdjudicationComponent i : adjudication)
          dst.adjudication.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof SubDetailComponent))
        return false;
      SubDetailComponent o = (SubDetailComponent) other_;
      return compareDeep(sequence, o.sequence, true) && compareDeep(revenue, o.revenue, true)
          && compareDeep(category, o.category, true) && compareDeep(productOrService, o.productOrService, true)
          && compareDeep(modifier, o.modifier, true) && compareDeep(programCode, o.programCode, true)
          && compareDeep(quantity, o.quantity, true) && compareDeep(unitPrice, o.unitPrice, true)
          && compareDeep(factor, o.factor, true) && compareDeep(net, o.net, true) && compareDeep(udi, o.udi, true)
          && compareDeep(noteNumber, o.noteNumber, true) && compareDeep(adjudication, o.adjudication, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof SubDetailComponent))
        return false;
      SubDetailComponent o = (SubDetailComponent) other_;
      return compareValues(sequence, o.sequence, true) && compareValues(factor, o.factor, true)
          && compareValues(noteNumber, o.noteNumber, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(sequence, revenue, category, productOrService,
          modifier, programCode, quantity, unitPrice, factor, net, udi, noteNumber, adjudication);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.item.detail.subDetail";

    }

  }

  @Block()
  public static class AddedItemComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Claim items which this service line is intended to replace.
     */
    @Child(name = "itemSequence", type = {
        PositiveIntType.class }, order = 1, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Item sequence number", formalDefinition = "Claim items which this service line is intended to replace.")
    protected List<PositiveIntType> itemSequence;

    /**
     * The sequence number of the details within the claim item which this line is
     * intended to replace.
     */
    @Child(name = "detailSequence", type = {
        PositiveIntType.class }, order = 2, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Detail sequence number", formalDefinition = "The sequence number of the details within the claim item which this line is intended to replace.")
    protected List<PositiveIntType> detailSequence;

    /**
     * The sequence number of the sub-details woithin the details within the claim
     * item which this line is intended to replace.
     */
    @Child(name = "subDetailSequence", type = {
        PositiveIntType.class }, order = 3, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Subdetail sequence number", formalDefinition = "The sequence number of the sub-details woithin the details within the claim item which this line is intended to replace.")
    protected List<PositiveIntType> subDetailSequence;

    /**
     * The providers who are authorized for the services rendered to the patient.
     */
    @Child(name = "provider", type = { Practitioner.class, PractitionerRole.class,
        Organization.class }, order = 4, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Authorized providers", formalDefinition = "The providers who are authorized for the services rendered to the patient.")
    protected List<Reference> provider;

    /**
     * When the value is a group code then this item collects a set of related claim
     * details, otherwise this contains the product, service, drug or other billing
     * code for the item.
     */
    @Child(name = "productOrService", type = {
        CodeableConcept.class }, order = 5, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Billing, service, product, or drug code", formalDefinition = "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-uscls")
    protected CodeableConcept productOrService;

    /**
     * Item typification or modifiers codes to convey additional context for the
     * product or service.
     */
    @Child(name = "modifier", type = {
        CodeableConcept.class }, order = 6, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Service/Product billing modifiers", formalDefinition = "Item typification or modifiers codes to convey additional context for the product or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-modifiers")
    protected List<CodeableConcept> modifier;

    /**
     * Identifies the program under which this may be recovered.
     */
    @Child(name = "programCode", type = {
        CodeableConcept.class }, order = 7, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Program the product or service is provided under", formalDefinition = "Identifies the program under which this may be recovered.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-program-code")
    protected List<CodeableConcept> programCode;

    /**
     * The date or dates when the service or product was supplied, performed or
     * completed.
     */
    @Child(name = "serviced", type = { DateType.class,
        Period.class }, order = 8, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Date or dates of service or product delivery", formalDefinition = "The date or dates when the service or product was supplied, performed or completed.")
    protected DataType serviced;

    /**
     * Where the product or service was provided.
     */
    @Child(name = "location", type = { CodeableConcept.class, Address.class,
        Location.class }, order = 9, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Place of service or where product was supplied", formalDefinition = "Where the product or service was provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-place")
    protected DataType location;

    /**
     * The number of repetitions of a service or product.
     */
    @Child(name = "quantity", type = {
        Quantity.class }, order = 10, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Count of products or services", formalDefinition = "The number of repetitions of a service or product.")
    protected Quantity quantity;

    /**
     * If the item is not a group then this is the fee for the product or service,
     * otherwise this is the total of the fees for the details of the group.
     */
    @Child(name = "unitPrice", type = { Money.class }, order = 11, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Fee, charge or cost per item", formalDefinition = "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.")
    protected Money unitPrice;

    /**
     * A real number that represents a multiplier used in determining the overall
     * value of services delivered and/or goods received. The concept of a Factor
     * allows for a discount or surcharge multiplier to be applied to a monetary
     * amount.
     */
    @Child(name = "factor", type = {
        DecimalType.class }, order = 12, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Price scaling factor", formalDefinition = "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.")
    protected DecimalType factor;

    /**
     * The quantity times the unit price for an additional service or product or
     * charge.
     */
    @Child(name = "net", type = { Money.class }, order = 13, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Total item cost", formalDefinition = "The quantity times the unit price for an additional service or product or charge.")
    protected Money net;

    /**
     * Physical service site on the patient (limb, tooth, etc.).
     */
    @Child(name = "bodySite", type = {
        CodeableConcept.class }, order = 14, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Anatomical location", formalDefinition = "Physical service site on the patient (limb, tooth, etc.).")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/tooth")
    protected CodeableConcept bodySite;

    /**
     * A region or surface of the bodySite, e.g. limb region or tooth surface(s).
     */
    @Child(name = "subSite", type = {
        CodeableConcept.class }, order = 15, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Anatomical sub-location", formalDefinition = "A region or surface of the bodySite, e.g. limb region or tooth surface(s).")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/surface")
    protected List<CodeableConcept> subSite;

    /**
     * The numbers associated with notes below which apply to the adjudication of
     * this item.
     */
    @Child(name = "noteNumber", type = {
        PositiveIntType.class }, order = 16, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable note numbers", formalDefinition = "The numbers associated with notes below which apply to the adjudication of this item.")
    protected List<PositiveIntType> noteNumber;

    /**
     * The adjudication results.
     */
    @Child(name = "adjudication", type = {
        AdjudicationComponent.class }, order = 17, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Added items adjudication", formalDefinition = "The adjudication results.")
    protected List<AdjudicationComponent> adjudication;

    /**
     * The second-tier service adjudications for payor added services.
     */
    @Child(name = "detail", type = {}, order = 18, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Insurer added line items", formalDefinition = "The second-tier service adjudications for payor added services.")
    protected List<AddedItemDetailComponent> detail;

    private static final long serialVersionUID = 2055842184L;

    /**
     * Constructor
     */
    public AddedItemComponent() {
      super();
    }

    /**
     * Constructor
     */
    public AddedItemComponent(CodeableConcept productOrService) {
      super();
      this.setProductOrService(productOrService);
    }

    /**
     * @return {@link #itemSequence} (Claim items which this service line is
     *         intended to replace.)
     */
    public List<PositiveIntType> getItemSequence() {
      if (this.itemSequence == null)
        this.itemSequence = new ArrayList<PositiveIntType>();
      return this.itemSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setItemSequence(List<PositiveIntType> theItemSequence) {
      this.itemSequence = theItemSequence;
      return this;
    }

    public boolean hasItemSequence() {
      if (this.itemSequence == null)
        return false;
      for (PositiveIntType item : this.itemSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #itemSequence} (Claim items which this service line is
     *         intended to replace.)
     */
    public PositiveIntType addItemSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.itemSequence == null)
        this.itemSequence = new ArrayList<PositiveIntType>();
      this.itemSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #itemSequence} (Claim items which this service line is
     *              intended to replace.)
     */
    public AddedItemComponent addItemSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.itemSequence == null)
        this.itemSequence = new ArrayList<PositiveIntType>();
      this.itemSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #itemSequence} (Claim items which this service line is
     *              intended to replace.)
     */
    public boolean hasItemSequence(int value) {
      if (this.itemSequence == null)
        return false;
      for (PositiveIntType v : this.itemSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #detailSequence} (The sequence number of the details within
     *         the claim item which this line is intended to replace.)
     */
    public List<PositiveIntType> getDetailSequence() {
      if (this.detailSequence == null)
        this.detailSequence = new ArrayList<PositiveIntType>();
      return this.detailSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setDetailSequence(List<PositiveIntType> theDetailSequence) {
      this.detailSequence = theDetailSequence;
      return this;
    }

    public boolean hasDetailSequence() {
      if (this.detailSequence == null)
        return false;
      for (PositiveIntType item : this.detailSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #detailSequence} (The sequence number of the details within
     *         the claim item which this line is intended to replace.)
     */
    public PositiveIntType addDetailSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.detailSequence == null)
        this.detailSequence = new ArrayList<PositiveIntType>();
      this.detailSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #detailSequence} (The sequence number of the details
     *              within the claim item which this line is intended to replace.)
     */
    public AddedItemComponent addDetailSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.detailSequence == null)
        this.detailSequence = new ArrayList<PositiveIntType>();
      this.detailSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #detailSequence} (The sequence number of the details
     *              within the claim item which this line is intended to replace.)
     */
    public boolean hasDetailSequence(int value) {
      if (this.detailSequence == null)
        return false;
      for (PositiveIntType v : this.detailSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #subDetailSequence} (The sequence number of the sub-details
     *         woithin the details within the claim item which this line is intended
     *         to replace.)
     */
    public List<PositiveIntType> getSubDetailSequence() {
      if (this.subDetailSequence == null)
        this.subDetailSequence = new ArrayList<PositiveIntType>();
      return this.subDetailSequence;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setSubDetailSequence(List<PositiveIntType> theSubDetailSequence) {
      this.subDetailSequence = theSubDetailSequence;
      return this;
    }

    public boolean hasSubDetailSequence() {
      if (this.subDetailSequence == null)
        return false;
      for (PositiveIntType item : this.subDetailSequence)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #subDetailSequence} (The sequence number of the sub-details
     *         woithin the details within the claim item which this line is intended
     *         to replace.)
     */
    public PositiveIntType addSubDetailSequenceElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.subDetailSequence == null)
        this.subDetailSequence = new ArrayList<PositiveIntType>();
      this.subDetailSequence.add(t);
      return t;
    }

    /**
     * @param value {@link #subDetailSequence} (The sequence number of the
     *              sub-details woithin the details within the claim item which this
     *              line is intended to replace.)
     */
    public AddedItemComponent addSubDetailSequence(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.subDetailSequence == null)
        this.subDetailSequence = new ArrayList<PositiveIntType>();
      this.subDetailSequence.add(t);
      return this;
    }

    /**
     * @param value {@link #subDetailSequence} (The sequence number of the
     *              sub-details woithin the details within the claim item which this
     *              line is intended to replace.)
     */
    public boolean hasSubDetailSequence(int value) {
      if (this.subDetailSequence == null)
        return false;
      for (PositiveIntType v : this.subDetailSequence)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #provider} (The providers who are authorized for the services
     *         rendered to the patient.)
     */
    public List<Reference> getProvider() {
      if (this.provider == null)
        this.provider = new ArrayList<Reference>();
      return this.provider;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setProvider(List<Reference> theProvider) {
      this.provider = theProvider;
      return this;
    }

    public boolean hasProvider() {
      if (this.provider == null)
        return false;
      for (Reference item : this.provider)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public Reference addProvider() { // 3
      Reference t = new Reference();
      if (this.provider == null)
        this.provider = new ArrayList<Reference>();
      this.provider.add(t);
      return t;
    }

    public AddedItemComponent addProvider(Reference t) { // 3
      if (t == null)
        return this;
      if (this.provider == null)
        this.provider = new ArrayList<Reference>();
      this.provider.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #provider}, creating
     *         it if it does not already exist {3}
     */
    public Reference getProviderFirstRep() {
      if (getProvider().isEmpty()) {
        addProvider();
      }
      return getProvider().get(0);
    }

    /**
     * @return {@link #productOrService} (When the value is a group code then this
     *         item collects a set of related claim details, otherwise this contains
     *         the product, service, drug or other billing code for the item.)
     */
    public CodeableConcept getProductOrService() {
      if (this.productOrService == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemComponent.productOrService");
        else if (Configuration.doAutoCreate())
          this.productOrService = new CodeableConcept(); // cc
      return this.productOrService;
    }

    public boolean hasProductOrService() {
      return this.productOrService != null && !this.productOrService.isEmpty();
    }

    /**
     * @param value {@link #productOrService} (When the value is a group code then
     *              this item collects a set of related claim details, otherwise
     *              this contains the product, service, drug or other billing code
     *              for the item.)
     */
    public AddedItemComponent setProductOrService(CodeableConcept value) {
      this.productOrService = value;
      return this;
    }

    /**
     * @return {@link #modifier} (Item typification or modifiers codes to convey
     *         additional context for the product or service.)
     */
    public List<CodeableConcept> getModifier() {
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      return this.modifier;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setModifier(List<CodeableConcept> theModifier) {
      this.modifier = theModifier;
      return this;
    }

    public boolean hasModifier() {
      if (this.modifier == null)
        return false;
      for (CodeableConcept item : this.modifier)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addModifier() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return t;
    }

    public AddedItemComponent addModifier(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #modifier}, creating
     *         it if it does not already exist {3}
     */
    public CodeableConcept getModifierFirstRep() {
      if (getModifier().isEmpty()) {
        addModifier();
      }
      return getModifier().get(0);
    }

    /**
     * @return {@link #programCode} (Identifies the program under which this may be
     *         recovered.)
     */
    public List<CodeableConcept> getProgramCode() {
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      return this.programCode;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setProgramCode(List<CodeableConcept> theProgramCode) {
      this.programCode = theProgramCode;
      return this;
    }

    public boolean hasProgramCode() {
      if (this.programCode == null)
        return false;
      for (CodeableConcept item : this.programCode)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addProgramCode() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return t;
    }

    public AddedItemComponent addProgramCode(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.programCode == null)
        this.programCode = new ArrayList<CodeableConcept>();
      this.programCode.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #programCode},
     *         creating it if it does not already exist {3}
     */
    public CodeableConcept getProgramCodeFirstRep() {
      if (getProgramCode().isEmpty()) {
        addProgramCode();
      }
      return getProgramCode().get(0);
    }

    /**
     * @return {@link #serviced} (The date or dates when the service or product was
     *         supplied, performed or completed.)
     */
    public DataType getServiced() {
      return this.serviced;
    }

    /**
     * @return {@link #serviced} (The date or dates when the service or product was
     *         supplied, performed or completed.)
     */
    public DateType getServicedDateType() throws FHIRException {
      if (this.serviced == null)
        this.serviced = new DateType();
      if (!(this.serviced instanceof DateType))
        throw new FHIRException("Type mismatch: the type DateType was expected, but "
            + this.serviced.getClass().getName() + " was encountered");
      return (DateType) this.serviced;
    }

    public boolean hasServicedDateType() {
        return this.serviced instanceof DateType;
    }

    /**
     * @return {@link #serviced} (The date or dates when the service or product was
     *         supplied, performed or completed.)
     */
    public Period getServicedPeriod() throws FHIRException {
      if (this.serviced == null)
        this.serviced = new Period();
      if (!(this.serviced instanceof Period))
        throw new FHIRException("Type mismatch: the type Period was expected, but " + this.serviced.getClass().getName()
            + " was encountered");
      return (Period) this.serviced;
    }

    public boolean hasServicedPeriod() {
        return this.serviced instanceof Period;
    }

    public boolean hasServiced() {
      return this.serviced != null && !this.serviced.isEmpty();
    }

    /**
     * @param value {@link #serviced} (The date or dates when the service or product
     *              was supplied, performed or completed.)
     */
    public AddedItemComponent setServiced(DataType value) {
      if (value != null && !(value instanceof DateType || value instanceof Period))
        throw new Error("Not the right type for ExplanationOfBenefit.addItem.serviced[x]: " + value.fhirType());
      this.serviced = value;
      return this;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public DataType getLocation() {
      return this.location;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public CodeableConcept getLocationCodeableConcept() throws FHIRException {
      if (this.location == null)
        this.location = new CodeableConcept();
      if (!(this.location instanceof CodeableConcept))
        throw new FHIRException("Type mismatch: the type CodeableConcept was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (CodeableConcept) this.location;
    }

    public boolean hasLocationCodeableConcept() {
        return this.location instanceof CodeableConcept;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public Address getLocationAddress() throws FHIRException {
      if (this.location == null)
        this.location = new Address();
      if (!(this.location instanceof Address))
        throw new FHIRException("Type mismatch: the type Address was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (Address) this.location;
    }

    public boolean hasLocationAddress() {
        return this.location instanceof Address;
    }

    /**
     * @return {@link #location} (Where the product or service was provided.)
     */
    public Reference getLocationReference() throws FHIRException {
      if (this.location == null)
        this.location = new Reference();
      if (!(this.location instanceof Reference))
        throw new FHIRException("Type mismatch: the type Reference was expected, but "
            + this.location.getClass().getName() + " was encountered");
      return (Reference) this.location;
    }

    public boolean hasLocationReference() {
        return this.location instanceof Reference;
    }

    public boolean hasLocation() {
      return this.location != null && !this.location.isEmpty();
    }

    /**
     * @param value {@link #location} (Where the product or service was provided.)
     */
    public AddedItemComponent setLocation(DataType value) {
      if (value != null
          && !(value instanceof CodeableConcept || value instanceof Address || value instanceof Reference))
        throw new Error("Not the right type for ExplanationOfBenefit.addItem.location[x]: " + value.fhirType());
      this.location = value;
      return this;
    }

    /**
     * @return {@link #quantity} (The number of repetitions of a service or
     *         product.)
     */
    public Quantity getQuantity() {
      if (this.quantity == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemComponent.quantity");
        else if (Configuration.doAutoCreate())
          this.quantity = new Quantity(); // cc
      return this.quantity;
    }

    public boolean hasQuantity() {
      return this.quantity != null && !this.quantity.isEmpty();
    }

    /**
     * @param value {@link #quantity} (The number of repetitions of a service or
     *              product.)
     */
    public AddedItemComponent setQuantity(Quantity value) {
      this.quantity = value;
      return this;
    }

    /**
     * @return {@link #unitPrice} (If the item is not a group then this is the fee
     *         for the product or service, otherwise this is the total of the fees
     *         for the details of the group.)
     */
    public Money getUnitPrice() {
      if (this.unitPrice == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemComponent.unitPrice");
        else if (Configuration.doAutoCreate())
          this.unitPrice = new Money(); // cc
      return this.unitPrice;
    }

    public boolean hasUnitPrice() {
      return this.unitPrice != null && !this.unitPrice.isEmpty();
    }

    /**
     * @param value {@link #unitPrice} (If the item is not a group then this is the
     *              fee for the product or service, otherwise this is the total of
     *              the fees for the details of the group.)
     */
    public AddedItemComponent setUnitPrice(Money value) {
      this.unitPrice = value;
      return this;
    }

    /**
     * @return {@link #factor} (A real number that represents a multiplier used in
     *         determining the overall value of services delivered and/or goods
     *         received. The concept of a Factor allows for a discount or surcharge
     *         multiplier to be applied to a monetary amount.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFactor" gives direct access to the value
     */
    public DecimalType getFactorElement() {
      if (this.factor == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemComponent.factor");
        else if (Configuration.doAutoCreate())
          this.factor = new DecimalType(); // bb
      return this.factor;
    }

    public boolean hasFactorElement() {
      return this.factor != null && !this.factor.isEmpty();
    }

    public boolean hasFactor() {
      return this.factor != null && !this.factor.isEmpty();
    }

    /**
     * @param value {@link #factor} (A real number that represents a multiplier used
     *              in determining the overall value of services delivered and/or
     *              goods received. The concept of a Factor allows for a discount or
     *              surcharge multiplier to be applied to a monetary amount.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getFactor" gives direct access to the value
     */
    public AddedItemComponent setFactorElement(DecimalType value) {
      this.factor = value;
      return this;
    }

    /**
     * @return A real number that represents a multiplier used in determining the
     *         overall value of services delivered and/or goods received. The
     *         concept of a Factor allows for a discount or surcharge multiplier to
     *         be applied to a monetary amount.
     */
    public BigDecimal getFactor() {
      return this.factor == null ? null : this.factor.getValue();
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemComponent setFactor(BigDecimal value) {
      if (value == null)
        this.factor = null;
      else {
        if (this.factor == null)
          this.factor = new DecimalType();
        this.factor.setValue(value);
      }
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemComponent setFactor(long value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemComponent setFactor(double value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @return {@link #net} (The quantity times the unit price for an additional
     *         service or product or charge.)
     */
    public Money getNet() {
      if (this.net == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemComponent.net");
        else if (Configuration.doAutoCreate())
          this.net = new Money(); // cc
      return this.net;
    }

    public boolean hasNet() {
      return this.net != null && !this.net.isEmpty();
    }

    /**
     * @param value {@link #net} (The quantity times the unit price for an
     *              additional service or product or charge.)
     */
    public AddedItemComponent setNet(Money value) {
      this.net = value;
      return this;
    }

    /**
     * @return {@link #bodySite} (Physical service site on the patient (limb, tooth,
     *         etc.).)
     */
    public CodeableConcept getBodySite() {
      if (this.bodySite == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemComponent.bodySite");
        else if (Configuration.doAutoCreate())
          this.bodySite = new CodeableConcept(); // cc
      return this.bodySite;
    }

    public boolean hasBodySite() {
      return this.bodySite != null && !this.bodySite.isEmpty();
    }

    /**
     * @param value {@link #bodySite} (Physical service site on the patient (limb,
     *              tooth, etc.).)
     */
    public AddedItemComponent setBodySite(CodeableConcept value) {
      this.bodySite = value;
      return this;
    }

    /**
     * @return {@link #subSite} (A region or surface of the bodySite, e.g. limb
     *         region or tooth surface(s).)
     */
    public List<CodeableConcept> getSubSite() {
      if (this.subSite == null)
        this.subSite = new ArrayList<CodeableConcept>();
      return this.subSite;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setSubSite(List<CodeableConcept> theSubSite) {
      this.subSite = theSubSite;
      return this;
    }

    public boolean hasSubSite() {
      if (this.subSite == null)
        return false;
      for (CodeableConcept item : this.subSite)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addSubSite() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.subSite == null)
        this.subSite = new ArrayList<CodeableConcept>();
      this.subSite.add(t);
      return t;
    }

    public AddedItemComponent addSubSite(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.subSite == null)
        this.subSite = new ArrayList<CodeableConcept>();
      this.subSite.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #subSite}, creating it
     *         if it does not already exist {3}
     */
    public CodeableConcept getSubSiteFirstRep() {
      if (getSubSite().isEmpty()) {
        addSubSite();
      }
      return getSubSite().get(0);
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public List<PositiveIntType> getNoteNumber() {
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      return this.noteNumber;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setNoteNumber(List<PositiveIntType> theNoteNumber) {
      this.noteNumber = theNoteNumber;
      return this;
    }

    public boolean hasNoteNumber() {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType item : this.noteNumber)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public PositiveIntType addNoteNumberElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return t;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public AddedItemComponent addNoteNumber(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return this;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public boolean hasNoteNumber(int value) {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType v : this.noteNumber)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #adjudication} (The adjudication results.)
     */
    public List<AdjudicationComponent> getAdjudication() {
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      return this.adjudication;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setAdjudication(List<AdjudicationComponent> theAdjudication) {
      this.adjudication = theAdjudication;
      return this;
    }

    public boolean hasAdjudication() {
      if (this.adjudication == null)
        return false;
      for (AdjudicationComponent item : this.adjudication)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AdjudicationComponent addAdjudication() { // 3
      AdjudicationComponent t = new AdjudicationComponent();
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return t;
    }

    public AddedItemComponent addAdjudication(AdjudicationComponent t) { // 3
      if (t == null)
        return this;
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #adjudication},
     *         creating it if it does not already exist {3}
     */
    public AdjudicationComponent getAdjudicationFirstRep() {
      if (getAdjudication().isEmpty()) {
        addAdjudication();
      }
      return getAdjudication().get(0);
    }

    /**
     * @return {@link #detail} (The second-tier service adjudications for payor
     *         added services.)
     */
    public List<AddedItemDetailComponent> getDetail() {
      if (this.detail == null)
        this.detail = new ArrayList<AddedItemDetailComponent>();
      return this.detail;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemComponent setDetail(List<AddedItemDetailComponent> theDetail) {
      this.detail = theDetail;
      return this;
    }

    public boolean hasDetail() {
      if (this.detail == null)
        return false;
      for (AddedItemDetailComponent item : this.detail)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AddedItemDetailComponent addDetail() { // 3
      AddedItemDetailComponent t = new AddedItemDetailComponent();
      if (this.detail == null)
        this.detail = new ArrayList<AddedItemDetailComponent>();
      this.detail.add(t);
      return t;
    }

    public AddedItemComponent addDetail(AddedItemDetailComponent t) { // 3
      if (t == null)
        return this;
      if (this.detail == null)
        this.detail = new ArrayList<AddedItemDetailComponent>();
      this.detail.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #detail}, creating it
     *         if it does not already exist {3}
     */
    public AddedItemDetailComponent getDetailFirstRep() {
      if (getDetail().isEmpty()) {
        addDetail();
      }
      return getDetail().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("itemSequence", "positiveInt",
          "Claim items which this service line is intended to replace.", 0, java.lang.Integer.MAX_VALUE, itemSequence));
      children.add(new Property("detailSequence", "positiveInt",
          "The sequence number of the details within the claim item which this line is intended to replace.", 0,
          java.lang.Integer.MAX_VALUE, detailSequence));
      children.add(new Property("subDetailSequence", "positiveInt",
          "The sequence number of the sub-details woithin the details within the claim item which this line is intended to replace.",
          0, java.lang.Integer.MAX_VALUE, subDetailSequence));
      children.add(new Property("provider", "Reference(Practitioner|PractitionerRole|Organization)",
          "The providers who are authorized for the services rendered to the patient.", 0, java.lang.Integer.MAX_VALUE,
          provider));
      children.add(new Property("productOrService", "CodeableConcept",
          "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
          0, 1, productOrService));
      children.add(new Property("modifier", "CodeableConcept",
          "Item typification or modifiers codes to convey additional context for the product or service.", 0,
          java.lang.Integer.MAX_VALUE, modifier));
      children.add(new Property("programCode", "CodeableConcept",
          "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode));
      children.add(new Property("serviced[x]", "date|Period",
          "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced));
      children.add(new Property("location[x]", "CodeableConcept|Address|Reference(Location)",
          "Where the product or service was provided.", 0, 1, location));
      children.add(
          new Property("quantity", "Quantity", "The number of repetitions of a service or product.", 0, 1, quantity));
      children.add(new Property("unitPrice", "Money",
          "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
          0, 1, unitPrice));
      children.add(new Property("factor", "decimal",
          "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
          0, 1, factor));
      children.add(new Property("net", "Money",
          "The quantity times the unit price for an additional service or product or charge.", 0, 1, net));
      children.add(new Property("bodySite", "CodeableConcept",
          "Physical service site on the patient (limb, tooth, etc.).", 0, 1, bodySite));
      children.add(new Property("subSite", "CodeableConcept",
          "A region or surface of the bodySite, e.g. limb region or tooth surface(s).", 0, java.lang.Integer.MAX_VALUE,
          subSite));
      children.add(new Property("noteNumber", "positiveInt",
          "The numbers associated with notes below which apply to the adjudication of this item.", 0,
          java.lang.Integer.MAX_VALUE, noteNumber));
      children.add(new Property("adjudication", "@ExplanationOfBenefit.item.adjudication", "The adjudication results.",
          0, java.lang.Integer.MAX_VALUE, adjudication));
      children.add(new Property("detail", "", "The second-tier service adjudications for payor added services.", 0,
          java.lang.Integer.MAX_VALUE, detail));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1977979892:
        /* itemSequence */ return new Property("itemSequence", "positiveInt",
            "Claim items which this service line is intended to replace.", 0, java.lang.Integer.MAX_VALUE,
            itemSequence);
      case 1321472818:
        /* detailSequence */ return new Property("detailSequence", "positiveInt",
            "The sequence number of the details within the claim item which this line is intended to replace.", 0,
            java.lang.Integer.MAX_VALUE, detailSequence);
      case -855462510:
        /* subDetailSequence */ return new Property("subDetailSequence", "positiveInt",
            "The sequence number of the sub-details woithin the details within the claim item which this line is intended to replace.",
            0, java.lang.Integer.MAX_VALUE, subDetailSequence);
      case -987494927:
        /* provider */ return new Property("provider", "Reference(Practitioner|PractitionerRole|Organization)",
            "The providers who are authorized for the services rendered to the patient.", 0,
            java.lang.Integer.MAX_VALUE, provider);
      case 1957227299:
        /* productOrService */ return new Property("productOrService", "CodeableConcept",
            "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
            0, 1, productOrService);
      case -615513385:
        /* modifier */ return new Property("modifier", "CodeableConcept",
            "Item typification or modifiers codes to convey additional context for the product or service.", 0,
            java.lang.Integer.MAX_VALUE, modifier);
      case 1010065041:
        /* programCode */ return new Property("programCode", "CodeableConcept",
            "Identifies the program under which this may be recovered.", 0, java.lang.Integer.MAX_VALUE, programCode);
      case -1927922223:
        /* serviced[x] */ return new Property("serviced[x]", "date|Period",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 1379209295:
        /* serviced */ return new Property("serviced[x]", "date|Period",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 363246749:
        /* servicedDate */ return new Property("serviced[x]", "date",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 1534966512:
        /* servicedPeriod */ return new Property("serviced[x]", "Period",
            "The date or dates when the service or product was supplied, performed or completed.", 0, 1, serviced);
      case 552316075:
        /* location[x] */ return new Property("location[x]", "CodeableConcept|Address|Reference(Location)",
            "Where the product or service was provided.", 0, 1, location);
      case 1901043637:
        /* location */ return new Property("location[x]", "CodeableConcept|Address|Reference(Location)",
            "Where the product or service was provided.", 0, 1, location);
      case -1224800468:
        /* locationCodeableConcept */ return new Property("location[x]", "CodeableConcept",
            "Where the product or service was provided.", 0, 1, location);
      case -1280020865:
        /* locationAddress */ return new Property("location[x]", "Address",
            "Where the product or service was provided.", 0, 1, location);
      case 755866390:
        /* locationReference */ return new Property("location[x]", "Reference(Location)",
            "Where the product or service was provided.", 0, 1, location);
      case -1285004149:
        /* quantity */ return new Property("quantity", "Quantity", "The number of repetitions of a service or product.",
            0, 1, quantity);
      case -486196699:
        /* unitPrice */ return new Property("unitPrice", "Money",
            "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
            0, 1, unitPrice);
      case -1282148017:
        /* factor */ return new Property("factor", "decimal",
            "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
            0, 1, factor);
      case 108957:
        /* net */ return new Property("net", "Money",
            "The quantity times the unit price for an additional service or product or charge.", 0, 1, net);
      case 1702620169:
        /* bodySite */ return new Property("bodySite", "CodeableConcept",
            "Physical service site on the patient (limb, tooth, etc.).", 0, 1, bodySite);
      case -1868566105:
        /* subSite */ return new Property("subSite", "CodeableConcept",
            "A region or surface of the bodySite, e.g. limb region or tooth surface(s).", 0,
            java.lang.Integer.MAX_VALUE, subSite);
      case -1110033957:
        /* noteNumber */ return new Property("noteNumber", "positiveInt",
            "The numbers associated with notes below which apply to the adjudication of this item.", 0,
            java.lang.Integer.MAX_VALUE, noteNumber);
      case -231349275:
        /* adjudication */ return new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
            "The adjudication results.", 0, java.lang.Integer.MAX_VALUE, adjudication);
      case -1335224239:
        /* detail */ return new Property("detail", "",
            "The second-tier service adjudications for payor added services.", 0, java.lang.Integer.MAX_VALUE, detail);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1977979892:
        /* itemSequence */ return this.itemSequence == null ? new Base[0]
            : this.itemSequence.toArray(new Base[this.itemSequence.size()]); // PositiveIntType
      case 1321472818:
        /* detailSequence */ return this.detailSequence == null ? new Base[0]
            : this.detailSequence.toArray(new Base[this.detailSequence.size()]); // PositiveIntType
      case -855462510:
        /* subDetailSequence */ return this.subDetailSequence == null ? new Base[0]
            : this.subDetailSequence.toArray(new Base[this.subDetailSequence.size()]); // PositiveIntType
      case -987494927:
        /* provider */ return this.provider == null ? new Base[0]
            : this.provider.toArray(new Base[this.provider.size()]); // Reference
      case 1957227299:
        /* productOrService */ return this.productOrService == null ? new Base[0]
            : new Base[] { this.productOrService }; // CodeableConcept
      case -615513385:
        /* modifier */ return this.modifier == null ? new Base[0]
            : this.modifier.toArray(new Base[this.modifier.size()]); // CodeableConcept
      case 1010065041:
        /* programCode */ return this.programCode == null ? new Base[0]
            : this.programCode.toArray(new Base[this.programCode.size()]); // CodeableConcept
      case 1379209295:
        /* serviced */ return this.serviced == null ? new Base[0] : new Base[] { this.serviced }; // DataType
      case 1901043637:
        /* location */ return this.location == null ? new Base[0] : new Base[] { this.location }; // DataType
      case -1285004149:
        /* quantity */ return this.quantity == null ? new Base[0] : new Base[] { this.quantity }; // Quantity
      case -486196699:
        /* unitPrice */ return this.unitPrice == null ? new Base[0] : new Base[] { this.unitPrice }; // Money
      case -1282148017:
        /* factor */ return this.factor == null ? new Base[0] : new Base[] { this.factor }; // DecimalType
      case 108957:
        /* net */ return this.net == null ? new Base[0] : new Base[] { this.net }; // Money
      case 1702620169:
        /* bodySite */ return this.bodySite == null ? new Base[0] : new Base[] { this.bodySite }; // CodeableConcept
      case -1868566105:
        /* subSite */ return this.subSite == null ? new Base[0] : this.subSite.toArray(new Base[this.subSite.size()]); // CodeableConcept
      case -1110033957:
        /* noteNumber */ return this.noteNumber == null ? new Base[0]
            : this.noteNumber.toArray(new Base[this.noteNumber.size()]); // PositiveIntType
      case -231349275:
        /* adjudication */ return this.adjudication == null ? new Base[0]
            : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
      case -1335224239:
        /* detail */ return this.detail == null ? new Base[0] : this.detail.toArray(new Base[this.detail.size()]); // AddedItemDetailComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1977979892: // itemSequence
        this.getItemSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case 1321472818: // detailSequence
        this.getDetailSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -855462510: // subDetailSequence
        this.getSubDetailSequence().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -987494927: // provider
        this.getProvider().add(TypeConvertor.castToReference(value)); // Reference
        return value;
      case 1957227299: // productOrService
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -615513385: // modifier
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1010065041: // programCode
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case 1379209295: // serviced
        this.serviced = TypeConvertor.castToType(value); // DataType
        return value;
      case 1901043637: // location
        this.location = TypeConvertor.castToType(value); // DataType
        return value;
      case -1285004149: // quantity
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
        return value;
      case -486196699: // unitPrice
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1282148017: // factor
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      case 108957: // net
        this.net = TypeConvertor.castToMoney(value); // Money
        return value;
      case 1702620169: // bodySite
        this.bodySite = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -1868566105: // subSite
        this.getSubSite().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case -1110033957: // noteNumber
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -231349275: // adjudication
        this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
        return value;
      case -1335224239: // detail
        this.getDetail().add((AddedItemDetailComponent) value); // AddedItemDetailComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("itemSequence")) {
        this.getItemSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("detailSequence")) {
        this.getDetailSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("subDetailSequence")) {
        this.getSubDetailSequence().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("provider")) {
        this.getProvider().add(TypeConvertor.castToReference(value));
      } else if (name.equals("productOrService")) {
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("modifier")) {
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("programCode")) {
        this.getProgramCode().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("serviced[x]")) {
        this.serviced = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("location[x]")) {
        this.location = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("quantity")) {
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
      } else if (name.equals("unitPrice")) {
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("factor")) {
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
      } else if (name.equals("net")) {
        this.net = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("bodySite")) {
        this.bodySite = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("subSite")) {
        this.getSubSite().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("adjudication")) {
        this.getAdjudication().add((AdjudicationComponent) value);
      } else if (name.equals("detail")) {
        this.getDetail().add((AddedItemDetailComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("itemSequence")) {
        this.getItemSequence().remove(value);
      } else if (name.equals("detailSequence")) {
        this.getDetailSequence().remove(value);
      } else if (name.equals("subDetailSequence")) {
        this.getSubDetailSequence().remove(value);
      } else if (name.equals("provider")) {
        this.getProvider().remove(value);
      } else if (name.equals("productOrService")) {
        this.productOrService = null;
      } else if (name.equals("modifier")) {
        this.getModifier().remove(value);
      } else if (name.equals("programCode")) {
        this.getProgramCode().remove(value);
      } else if (name.equals("serviced[x]")) {
        this.serviced = null;
      } else if (name.equals("location[x]")) {
        this.location = null;
      } else if (name.equals("quantity")) {
        this.quantity = null;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = null;
      } else if (name.equals("factor")) {
        this.factor = null;
      } else if (name.equals("net")) {
        this.net = null;
      } else if (name.equals("bodySite")) {
        this.bodySite = null;
      } else if (name.equals("subSite")) {
        this.getSubSite().remove(value);
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().remove(value);
      } else if (name.equals("adjudication")) {
        this.getAdjudication().remove((AdjudicationComponent) value);
      } else if (name.equals("detail")) {
        this.getDetail().remove((AddedItemDetailComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1977979892:
        return addItemSequenceElement();
      case 1321472818:
        return addDetailSequenceElement();
      case -855462510:
        return addSubDetailSequenceElement();
      case -987494927:
        return addProvider();
      case 1957227299:
        return getProductOrService();
      case -615513385:
        return addModifier();
      case 1010065041:
        return addProgramCode();
      case -1927922223:
        return getServiced();
      case 1379209295:
        return getServiced();
      case 552316075:
        return getLocation();
      case 1901043637:
        return getLocation();
      case -1285004149:
        return getQuantity();
      case -486196699:
        return getUnitPrice();
      case -1282148017:
        return getFactorElement();
      case 108957:
        return getNet();
      case 1702620169:
        return getBodySite();
      case -1868566105:
        return addSubSite();
      case -1110033957:
        return addNoteNumberElement();
      case -231349275:
        return addAdjudication();
      case -1335224239:
        return addDetail();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1977979892:
        /* itemSequence */ return new String[] { "positiveInt" };
      case 1321472818:
        /* detailSequence */ return new String[] { "positiveInt" };
      case -855462510:
        /* subDetailSequence */ return new String[] { "positiveInt" };
      case -987494927:
        /* provider */ return new String[] { "Reference" };
      case 1957227299:
        /* productOrService */ return new String[] { "CodeableConcept" };
      case -615513385:
        /* modifier */ return new String[] { "CodeableConcept" };
      case 1010065041:
        /* programCode */ return new String[] { "CodeableConcept" };
      case 1379209295:
        /* serviced */ return new String[] { "date", "Period" };
      case 1901043637:
        /* location */ return new String[] { "CodeableConcept", "Address", "Reference" };
      case -1285004149:
        /* quantity */ return new String[] { "Quantity" };
      case -486196699:
        /* unitPrice */ return new String[] { "Money" };
      case -1282148017:
        /* factor */ return new String[] { "decimal" };
      case 108957:
        /* net */ return new String[] { "Money" };
      case 1702620169:
        /* bodySite */ return new String[] { "CodeableConcept" };
      case -1868566105:
        /* subSite */ return new String[] { "CodeableConcept" };
      case -1110033957:
        /* noteNumber */ return new String[] { "positiveInt" };
      case -231349275:
        /* adjudication */ return new String[] { "@ExplanationOfBenefit.item.adjudication" };
      case -1335224239:
        /* detail */ return new String[] {};
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("itemSequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.itemSequence");
      } else if (name.equals("detailSequence")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.detailSequence");
      } else if (name.equals("subDetailSequence")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.subDetailSequence");
      } else if (name.equals("provider")) {
        return addProvider();
      } else if (name.equals("productOrService")) {
        this.productOrService = new CodeableConcept();
        return this.productOrService;
      } else if (name.equals("modifier")) {
        return addModifier();
      } else if (name.equals("programCode")) {
        return addProgramCode();
      } else if (name.equals("servicedDate")) {
        this.serviced = new DateType();
        return this.serviced;
      } else if (name.equals("servicedPeriod")) {
        this.serviced = new Period();
        return this.serviced;
      } else if (name.equals("locationCodeableConcept")) {
        this.location = new CodeableConcept();
        return this.location;
      } else if (name.equals("locationAddress")) {
        this.location = new Address();
        return this.location;
      } else if (name.equals("locationReference")) {
        this.location = new Reference();
        return this.location;
      } else if (name.equals("quantity")) {
        this.quantity = new Quantity();
        return this.quantity;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = new Money();
        return this.unitPrice;
      } else if (name.equals("factor")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.factor");
      } else if (name.equals("net")) {
        this.net = new Money();
        return this.net;
      } else if (name.equals("bodySite")) {
        this.bodySite = new CodeableConcept();
        return this.bodySite;
      } else if (name.equals("subSite")) {
        return addSubSite();
      } else if (name.equals("noteNumber")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.noteNumber");
      } else if (name.equals("adjudication")) {
        return addAdjudication();
      } else if (name.equals("detail")) {
        return addDetail();
      } else
        return super.addChild(name);
    }

    public AddedItemComponent copy() {
      AddedItemComponent dst = new AddedItemComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(AddedItemComponent dst) {
      super.copyValues(dst);
      if (itemSequence != null) {
        dst.itemSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : itemSequence)
          dst.itemSequence.add(i.copy());
      }
      ;
      if (detailSequence != null) {
        dst.detailSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : detailSequence)
          dst.detailSequence.add(i.copy());
      }
      ;
      if (subDetailSequence != null) {
        dst.subDetailSequence = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : subDetailSequence)
          dst.subDetailSequence.add(i.copy());
      }
      ;
      if (provider != null) {
        dst.provider = new ArrayList<Reference>();
        for (Reference i : provider)
          dst.provider.add(i.copy());
      }
      ;
      dst.productOrService = productOrService == null ? null : productOrService.copy();
      if (modifier != null) {
        dst.modifier = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : modifier)
          dst.modifier.add(i.copy());
      }
      ;
      if (programCode != null) {
        dst.programCode = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : programCode)
          dst.programCode.add(i.copy());
      }
      ;
      dst.serviced = serviced == null ? null : serviced.copy();
      dst.location = location == null ? null : location.copy();
      dst.quantity = quantity == null ? null : quantity.copy();
      dst.unitPrice = unitPrice == null ? null : unitPrice.copy();
      dst.factor = factor == null ? null : factor.copy();
      dst.net = net == null ? null : net.copy();
      dst.bodySite = bodySite == null ? null : bodySite.copy();
      if (subSite != null) {
        dst.subSite = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : subSite)
          dst.subSite.add(i.copy());
      }
      ;
      if (noteNumber != null) {
        dst.noteNumber = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : noteNumber)
          dst.noteNumber.add(i.copy());
      }
      ;
      if (adjudication != null) {
        dst.adjudication = new ArrayList<AdjudicationComponent>();
        for (AdjudicationComponent i : adjudication)
          dst.adjudication.add(i.copy());
      }
      ;
      if (detail != null) {
        dst.detail = new ArrayList<AddedItemDetailComponent>();
        for (AddedItemDetailComponent i : detail)
          dst.detail.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof AddedItemComponent))
        return false;
      AddedItemComponent o = (AddedItemComponent) other_;
      return compareDeep(itemSequence, o.itemSequence, true) && compareDeep(detailSequence, o.detailSequence, true)
          && compareDeep(subDetailSequence, o.subDetailSequence, true) && compareDeep(provider, o.provider, true)
          && compareDeep(productOrService, o.productOrService, true) && compareDeep(modifier, o.modifier, true)
          && compareDeep(programCode, o.programCode, true) && compareDeep(serviced, o.serviced, true)
          && compareDeep(location, o.location, true) && compareDeep(quantity, o.quantity, true)
          && compareDeep(unitPrice, o.unitPrice, true) && compareDeep(factor, o.factor, true)
          && compareDeep(net, o.net, true) && compareDeep(bodySite, o.bodySite, true)
          && compareDeep(subSite, o.subSite, true) && compareDeep(noteNumber, o.noteNumber, true)
          && compareDeep(adjudication, o.adjudication, true) && compareDeep(detail, o.detail, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof AddedItemComponent))
        return false;
      AddedItemComponent o = (AddedItemComponent) other_;
      return compareValues(itemSequence, o.itemSequence, true) && compareValues(detailSequence, o.detailSequence, true)
          && compareValues(subDetailSequence, o.subDetailSequence, true) && compareValues(factor, o.factor, true)
          && compareValues(noteNumber, o.noteNumber, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(itemSequence, detailSequence, subDetailSequence,
          provider, productOrService, modifier, programCode, serviced, location, quantity, unitPrice, factor, net,
          bodySite, subSite, noteNumber, adjudication, detail);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.addItem";

    }

  }

  @Block()
  public static class AddedItemDetailComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * When the value is a group code then this item collects a set of related claim
     * details, otherwise this contains the product, service, drug or other billing
     * code for the item.
     */
    @Child(name = "productOrService", type = {
        CodeableConcept.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Billing, service, product, or drug code", formalDefinition = "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-uscls")
    protected CodeableConcept productOrService;

    /**
     * Item typification or modifiers codes to convey additional context for the
     * product or service.
     */
    @Child(name = "modifier", type = {
        CodeableConcept.class }, order = 2, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Service/Product billing modifiers", formalDefinition = "Item typification or modifiers codes to convey additional context for the product or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-modifiers")
    protected List<CodeableConcept> modifier;

    /**
     * The number of repetitions of a service or product.
     */
    @Child(name = "quantity", type = { Quantity.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Count of products or services", formalDefinition = "The number of repetitions of a service or product.")
    protected Quantity quantity;

    /**
     * If the item is not a group then this is the fee for the product or service,
     * otherwise this is the total of the fees for the details of the group.
     */
    @Child(name = "unitPrice", type = { Money.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Fee, charge or cost per item", formalDefinition = "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.")
    protected Money unitPrice;

    /**
     * A real number that represents a multiplier used in determining the overall
     * value of services delivered and/or goods received. The concept of a Factor
     * allows for a discount or surcharge multiplier to be applied to a monetary
     * amount.
     */
    @Child(name = "factor", type = {
        DecimalType.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Price scaling factor", formalDefinition = "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.")
    protected DecimalType factor;

    /**
     * The quantity times the unit price for an additional service or product or
     * charge.
     */
    @Child(name = "net", type = { Money.class }, order = 6, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Total item cost", formalDefinition = "The quantity times the unit price for an additional service or product or charge.")
    protected Money net;

    /**
     * The numbers associated with notes below which apply to the adjudication of
     * this item.
     */
    @Child(name = "noteNumber", type = {
        PositiveIntType.class }, order = 7, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable note numbers", formalDefinition = "The numbers associated with notes below which apply to the adjudication of this item.")
    protected List<PositiveIntType> noteNumber;

    /**
     * The adjudication results.
     */
    @Child(name = "adjudication", type = {
        AdjudicationComponent.class }, order = 8, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Added items adjudication", formalDefinition = "The adjudication results.")
    protected List<AdjudicationComponent> adjudication;

    /**
     * The third-tier service adjudications for payor added services.
     */
    @Child(name = "subDetail", type = {}, order = 9, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Insurer added line items", formalDefinition = "The third-tier service adjudications for payor added services.")
    protected List<AddedItemDetailSubDetailComponent> subDetail;

    private static final long serialVersionUID = 295910869L;

    /**
     * Constructor
     */
    public AddedItemDetailComponent() {
      super();
    }

    /**
     * Constructor
     */
    public AddedItemDetailComponent(CodeableConcept productOrService) {
      super();
      this.setProductOrService(productOrService);
    }

    /**
     * @return {@link #productOrService} (When the value is a group code then this
     *         item collects a set of related claim details, otherwise this contains
     *         the product, service, drug or other billing code for the item.)
     */
    public CodeableConcept getProductOrService() {
      if (this.productOrService == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailComponent.productOrService");
        else if (Configuration.doAutoCreate())
          this.productOrService = new CodeableConcept(); // cc
      return this.productOrService;
    }

    public boolean hasProductOrService() {
      return this.productOrService != null && !this.productOrService.isEmpty();
    }

    /**
     * @param value {@link #productOrService} (When the value is a group code then
     *              this item collects a set of related claim details, otherwise
     *              this contains the product, service, drug or other billing code
     *              for the item.)
     */
    public AddedItemDetailComponent setProductOrService(CodeableConcept value) {
      this.productOrService = value;
      return this;
    }

    /**
     * @return {@link #modifier} (Item typification or modifiers codes to convey
     *         additional context for the product or service.)
     */
    public List<CodeableConcept> getModifier() {
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      return this.modifier;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailComponent setModifier(List<CodeableConcept> theModifier) {
      this.modifier = theModifier;
      return this;
    }

    public boolean hasModifier() {
      if (this.modifier == null)
        return false;
      for (CodeableConcept item : this.modifier)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addModifier() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return t;
    }

    public AddedItemDetailComponent addModifier(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #modifier}, creating
     *         it if it does not already exist {3}
     */
    public CodeableConcept getModifierFirstRep() {
      if (getModifier().isEmpty()) {
        addModifier();
      }
      return getModifier().get(0);
    }

    /**
     * @return {@link #quantity} (The number of repetitions of a service or
     *         product.)
     */
    public Quantity getQuantity() {
      if (this.quantity == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailComponent.quantity");
        else if (Configuration.doAutoCreate())
          this.quantity = new Quantity(); // cc
      return this.quantity;
    }

    public boolean hasQuantity() {
      return this.quantity != null && !this.quantity.isEmpty();
    }

    /**
     * @param value {@link #quantity} (The number of repetitions of a service or
     *              product.)
     */
    public AddedItemDetailComponent setQuantity(Quantity value) {
      this.quantity = value;
      return this;
    }

    /**
     * @return {@link #unitPrice} (If the item is not a group then this is the fee
     *         for the product or service, otherwise this is the total of the fees
     *         for the details of the group.)
     */
    public Money getUnitPrice() {
      if (this.unitPrice == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailComponent.unitPrice");
        else if (Configuration.doAutoCreate())
          this.unitPrice = new Money(); // cc
      return this.unitPrice;
    }

    public boolean hasUnitPrice() {
      return this.unitPrice != null && !this.unitPrice.isEmpty();
    }

    /**
     * @param value {@link #unitPrice} (If the item is not a group then this is the
     *              fee for the product or service, otherwise this is the total of
     *              the fees for the details of the group.)
     */
    public AddedItemDetailComponent setUnitPrice(Money value) {
      this.unitPrice = value;
      return this;
    }

    /**
     * @return {@link #factor} (A real number that represents a multiplier used in
     *         determining the overall value of services delivered and/or goods
     *         received. The concept of a Factor allows for a discount or surcharge
     *         multiplier to be applied to a monetary amount.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFactor" gives direct access to the value
     */
    public DecimalType getFactorElement() {
      if (this.factor == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailComponent.factor");
        else if (Configuration.doAutoCreate())
          this.factor = new DecimalType(); // bb
      return this.factor;
    }

    public boolean hasFactorElement() {
      return this.factor != null && !this.factor.isEmpty();
    }

    public boolean hasFactor() {
      return this.factor != null && !this.factor.isEmpty();
    }

    /**
     * @param value {@link #factor} (A real number that represents a multiplier used
     *              in determining the overall value of services delivered and/or
     *              goods received. The concept of a Factor allows for a discount or
     *              surcharge multiplier to be applied to a monetary amount.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getFactor" gives direct access to the value
     */
    public AddedItemDetailComponent setFactorElement(DecimalType value) {
      this.factor = value;
      return this;
    }

    /**
     * @return A real number that represents a multiplier used in determining the
     *         overall value of services delivered and/or goods received. The
     *         concept of a Factor allows for a discount or surcharge multiplier to
     *         be applied to a monetary amount.
     */
    public BigDecimal getFactor() {
      return this.factor == null ? null : this.factor.getValue();
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemDetailComponent setFactor(BigDecimal value) {
      if (value == null)
        this.factor = null;
      else {
        if (this.factor == null)
          this.factor = new DecimalType();
        this.factor.setValue(value);
      }
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemDetailComponent setFactor(long value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemDetailComponent setFactor(double value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @return {@link #net} (The quantity times the unit price for an additional
     *         service or product or charge.)
     */
    public Money getNet() {
      if (this.net == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailComponent.net");
        else if (Configuration.doAutoCreate())
          this.net = new Money(); // cc
      return this.net;
    }

    public boolean hasNet() {
      return this.net != null && !this.net.isEmpty();
    }

    /**
     * @param value {@link #net} (The quantity times the unit price for an
     *              additional service or product or charge.)
     */
    public AddedItemDetailComponent setNet(Money value) {
      this.net = value;
      return this;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public List<PositiveIntType> getNoteNumber() {
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      return this.noteNumber;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailComponent setNoteNumber(List<PositiveIntType> theNoteNumber) {
      this.noteNumber = theNoteNumber;
      return this;
    }

    public boolean hasNoteNumber() {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType item : this.noteNumber)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public PositiveIntType addNoteNumberElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return t;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public AddedItemDetailComponent addNoteNumber(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return this;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public boolean hasNoteNumber(int value) {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType v : this.noteNumber)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #adjudication} (The adjudication results.)
     */
    public List<AdjudicationComponent> getAdjudication() {
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      return this.adjudication;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailComponent setAdjudication(List<AdjudicationComponent> theAdjudication) {
      this.adjudication = theAdjudication;
      return this;
    }

    public boolean hasAdjudication() {
      if (this.adjudication == null)
        return false;
      for (AdjudicationComponent item : this.adjudication)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AdjudicationComponent addAdjudication() { // 3
      AdjudicationComponent t = new AdjudicationComponent();
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return t;
    }

    public AddedItemDetailComponent addAdjudication(AdjudicationComponent t) { // 3
      if (t == null)
        return this;
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #adjudication},
     *         creating it if it does not already exist {3}
     */
    public AdjudicationComponent getAdjudicationFirstRep() {
      if (getAdjudication().isEmpty()) {
        addAdjudication();
      }
      return getAdjudication().get(0);
    }

    /**
     * @return {@link #subDetail} (The third-tier service adjudications for payor
     *         added services.)
     */
    public List<AddedItemDetailSubDetailComponent> getSubDetail() {
      if (this.subDetail == null)
        this.subDetail = new ArrayList<AddedItemDetailSubDetailComponent>();
      return this.subDetail;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailComponent setSubDetail(List<AddedItemDetailSubDetailComponent> theSubDetail) {
      this.subDetail = theSubDetail;
      return this;
    }

    public boolean hasSubDetail() {
      if (this.subDetail == null)
        return false;
      for (AddedItemDetailSubDetailComponent item : this.subDetail)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AddedItemDetailSubDetailComponent addSubDetail() { // 3
      AddedItemDetailSubDetailComponent t = new AddedItemDetailSubDetailComponent();
      if (this.subDetail == null)
        this.subDetail = new ArrayList<AddedItemDetailSubDetailComponent>();
      this.subDetail.add(t);
      return t;
    }

    public AddedItemDetailComponent addSubDetail(AddedItemDetailSubDetailComponent t) { // 3
      if (t == null)
        return this;
      if (this.subDetail == null)
        this.subDetail = new ArrayList<AddedItemDetailSubDetailComponent>();
      this.subDetail.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #subDetail}, creating
     *         it if it does not already exist {3}
     */
    public AddedItemDetailSubDetailComponent getSubDetailFirstRep() {
      if (getSubDetail().isEmpty()) {
        addSubDetail();
      }
      return getSubDetail().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("productOrService", "CodeableConcept",
          "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
          0, 1, productOrService));
      children.add(new Property("modifier", "CodeableConcept",
          "Item typification or modifiers codes to convey additional context for the product or service.", 0,
          java.lang.Integer.MAX_VALUE, modifier));
      children.add(
          new Property("quantity", "Quantity", "The number of repetitions of a service or product.", 0, 1, quantity));
      children.add(new Property("unitPrice", "Money",
          "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
          0, 1, unitPrice));
      children.add(new Property("factor", "decimal",
          "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
          0, 1, factor));
      children.add(new Property("net", "Money",
          "The quantity times the unit price for an additional service or product or charge.", 0, 1, net));
      children.add(new Property("noteNumber", "positiveInt",
          "The numbers associated with notes below which apply to the adjudication of this item.", 0,
          java.lang.Integer.MAX_VALUE, noteNumber));
      children.add(new Property("adjudication", "@ExplanationOfBenefit.item.adjudication", "The adjudication results.",
          0, java.lang.Integer.MAX_VALUE, adjudication));
      children.add(new Property("subDetail", "", "The third-tier service adjudications for payor added services.", 0,
          java.lang.Integer.MAX_VALUE, subDetail));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1957227299:
        /* productOrService */ return new Property("productOrService", "CodeableConcept",
            "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
            0, 1, productOrService);
      case -615513385:
        /* modifier */ return new Property("modifier", "CodeableConcept",
            "Item typification or modifiers codes to convey additional context for the product or service.", 0,
            java.lang.Integer.MAX_VALUE, modifier);
      case -1285004149:
        /* quantity */ return new Property("quantity", "Quantity", "The number of repetitions of a service or product.",
            0, 1, quantity);
      case -486196699:
        /* unitPrice */ return new Property("unitPrice", "Money",
            "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
            0, 1, unitPrice);
      case -1282148017:
        /* factor */ return new Property("factor", "decimal",
            "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
            0, 1, factor);
      case 108957:
        /* net */ return new Property("net", "Money",
            "The quantity times the unit price for an additional service or product or charge.", 0, 1, net);
      case -1110033957:
        /* noteNumber */ return new Property("noteNumber", "positiveInt",
            "The numbers associated with notes below which apply to the adjudication of this item.", 0,
            java.lang.Integer.MAX_VALUE, noteNumber);
      case -231349275:
        /* adjudication */ return new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
            "The adjudication results.", 0, java.lang.Integer.MAX_VALUE, adjudication);
      case -828829007:
        /* subDetail */ return new Property("subDetail", "",
            "The third-tier service adjudications for payor added services.", 0, java.lang.Integer.MAX_VALUE,
            subDetail);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1957227299:
        /* productOrService */ return this.productOrService == null ? new Base[0]
            : new Base[] { this.productOrService }; // CodeableConcept
      case -615513385:
        /* modifier */ return this.modifier == null ? new Base[0]
            : this.modifier.toArray(new Base[this.modifier.size()]); // CodeableConcept
      case -1285004149:
        /* quantity */ return this.quantity == null ? new Base[0] : new Base[] { this.quantity }; // Quantity
      case -486196699:
        /* unitPrice */ return this.unitPrice == null ? new Base[0] : new Base[] { this.unitPrice }; // Money
      case -1282148017:
        /* factor */ return this.factor == null ? new Base[0] : new Base[] { this.factor }; // DecimalType
      case 108957:
        /* net */ return this.net == null ? new Base[0] : new Base[] { this.net }; // Money
      case -1110033957:
        /* noteNumber */ return this.noteNumber == null ? new Base[0]
            : this.noteNumber.toArray(new Base[this.noteNumber.size()]); // PositiveIntType
      case -231349275:
        /* adjudication */ return this.adjudication == null ? new Base[0]
            : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
      case -828829007:
        /* subDetail */ return this.subDetail == null ? new Base[0]
            : this.subDetail.toArray(new Base[this.subDetail.size()]); // AddedItemDetailSubDetailComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1957227299: // productOrService
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -615513385: // modifier
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case -1285004149: // quantity
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
        return value;
      case -486196699: // unitPrice
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1282148017: // factor
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      case 108957: // net
        this.net = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1110033957: // noteNumber
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -231349275: // adjudication
        this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
        return value;
      case -828829007: // subDetail
        this.getSubDetail().add((AddedItemDetailSubDetailComponent) value); // AddedItemDetailSubDetailComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("productOrService")) {
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("modifier")) {
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("quantity")) {
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
      } else if (name.equals("unitPrice")) {
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("factor")) {
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
      } else if (name.equals("net")) {
        this.net = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("adjudication")) {
        this.getAdjudication().add((AdjudicationComponent) value);
      } else if (name.equals("subDetail")) {
        this.getSubDetail().add((AddedItemDetailSubDetailComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("productOrService")) {
        this.productOrService = null;
      } else if (name.equals("modifier")) {
        this.getModifier().remove(value);
      } else if (name.equals("quantity")) {
        this.quantity = null;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = null;
      } else if (name.equals("factor")) {
        this.factor = null;
      } else if (name.equals("net")) {
        this.net = null;
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().remove(value);
      } else if (name.equals("adjudication")) {
        this.getAdjudication().remove((AdjudicationComponent) value);
      } else if (name.equals("subDetail")) {
        this.getSubDetail().remove((AddedItemDetailSubDetailComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1957227299:
        return getProductOrService();
      case -615513385:
        return addModifier();
      case -1285004149:
        return getQuantity();
      case -486196699:
        return getUnitPrice();
      case -1282148017:
        return getFactorElement();
      case 108957:
        return getNet();
      case -1110033957:
        return addNoteNumberElement();
      case -231349275:
        return addAdjudication();
      case -828829007:
        return addSubDetail();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1957227299:
        /* productOrService */ return new String[] { "CodeableConcept" };
      case -615513385:
        /* modifier */ return new String[] { "CodeableConcept" };
      case -1285004149:
        /* quantity */ return new String[] { "Quantity" };
      case -486196699:
        /* unitPrice */ return new String[] { "Money" };
      case -1282148017:
        /* factor */ return new String[] { "decimal" };
      case 108957:
        /* net */ return new String[] { "Money" };
      case -1110033957:
        /* noteNumber */ return new String[] { "positiveInt" };
      case -231349275:
        /* adjudication */ return new String[] { "@ExplanationOfBenefit.item.adjudication" };
      case -828829007:
        /* subDetail */ return new String[] {};
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("productOrService")) {
        this.productOrService = new CodeableConcept();
        return this.productOrService;
      } else if (name.equals("modifier")) {
        return addModifier();
      } else if (name.equals("quantity")) {
        this.quantity = new Quantity();
        return this.quantity;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = new Money();
        return this.unitPrice;
      } else if (name.equals("factor")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.detail.factor");
      } else if (name.equals("net")) {
        this.net = new Money();
        return this.net;
      } else if (name.equals("noteNumber")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.detail.noteNumber");
      } else if (name.equals("adjudication")) {
        return addAdjudication();
      } else if (name.equals("subDetail")) {
        return addSubDetail();
      } else
        return super.addChild(name);
    }

    public AddedItemDetailComponent copy() {
      AddedItemDetailComponent dst = new AddedItemDetailComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(AddedItemDetailComponent dst) {
      super.copyValues(dst);
      dst.productOrService = productOrService == null ? null : productOrService.copy();
      if (modifier != null) {
        dst.modifier = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : modifier)
          dst.modifier.add(i.copy());
      }
      ;
      dst.quantity = quantity == null ? null : quantity.copy();
      dst.unitPrice = unitPrice == null ? null : unitPrice.copy();
      dst.factor = factor == null ? null : factor.copy();
      dst.net = net == null ? null : net.copy();
      if (noteNumber != null) {
        dst.noteNumber = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : noteNumber)
          dst.noteNumber.add(i.copy());
      }
      ;
      if (adjudication != null) {
        dst.adjudication = new ArrayList<AdjudicationComponent>();
        for (AdjudicationComponent i : adjudication)
          dst.adjudication.add(i.copy());
      }
      ;
      if (subDetail != null) {
        dst.subDetail = new ArrayList<AddedItemDetailSubDetailComponent>();
        for (AddedItemDetailSubDetailComponent i : subDetail)
          dst.subDetail.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof AddedItemDetailComponent))
        return false;
      AddedItemDetailComponent o = (AddedItemDetailComponent) other_;
      return compareDeep(productOrService, o.productOrService, true) && compareDeep(modifier, o.modifier, true)
          && compareDeep(quantity, o.quantity, true) && compareDeep(unitPrice, o.unitPrice, true)
          && compareDeep(factor, o.factor, true) && compareDeep(net, o.net, true)
          && compareDeep(noteNumber, o.noteNumber, true) && compareDeep(adjudication, o.adjudication, true)
          && compareDeep(subDetail, o.subDetail, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof AddedItemDetailComponent))
        return false;
      AddedItemDetailComponent o = (AddedItemDetailComponent) other_;
      return compareValues(factor, o.factor, true) && compareValues(noteNumber, o.noteNumber, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(productOrService, modifier, quantity, unitPrice,
          factor, net, noteNumber, adjudication, subDetail);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.addItem.detail";

    }

  }

  @Block()
  public static class AddedItemDetailSubDetailComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * When the value is a group code then this item collects a set of related claim
     * details, otherwise this contains the product, service, drug or other billing
     * code for the item.
     */
    @Child(name = "productOrService", type = {
        CodeableConcept.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Billing, service, product, or drug code", formalDefinition = "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/service-uscls")
    protected CodeableConcept productOrService;

    /**
     * Item typification or modifiers codes to convey additional context for the
     * product or service.
     */
    @Child(name = "modifier", type = {
        CodeableConcept.class }, order = 2, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Service/Product billing modifiers", formalDefinition = "Item typification or modifiers codes to convey additional context for the product or service.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-modifiers")
    protected List<CodeableConcept> modifier;

    /**
     * The number of repetitions of a service or product.
     */
    @Child(name = "quantity", type = { Quantity.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Count of products or services", formalDefinition = "The number of repetitions of a service or product.")
    protected Quantity quantity;

    /**
     * If the item is not a group then this is the fee for the product or service,
     * otherwise this is the total of the fees for the details of the group.
     */
    @Child(name = "unitPrice", type = { Money.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Fee, charge or cost per item", formalDefinition = "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.")
    protected Money unitPrice;

    /**
     * A real number that represents a multiplier used in determining the overall
     * value of services delivered and/or goods received. The concept of a Factor
     * allows for a discount or surcharge multiplier to be applied to a monetary
     * amount.
     */
    @Child(name = "factor", type = {
        DecimalType.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Price scaling factor", formalDefinition = "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.")
    protected DecimalType factor;

    /**
     * The quantity times the unit price for an additional service or product or
     * charge.
     */
    @Child(name = "net", type = { Money.class }, order = 6, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Total item cost", formalDefinition = "The quantity times the unit price for an additional service or product or charge.")
    protected Money net;

    /**
     * The numbers associated with notes below which apply to the adjudication of
     * this item.
     */
    @Child(name = "noteNumber", type = {
        PositiveIntType.class }, order = 7, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Applicable note numbers", formalDefinition = "The numbers associated with notes below which apply to the adjudication of this item.")
    protected List<PositiveIntType> noteNumber;

    /**
     * The adjudication results.
     */
    @Child(name = "adjudication", type = {
        AdjudicationComponent.class }, order = 8, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Added items adjudication", formalDefinition = "The adjudication results.")
    protected List<AdjudicationComponent> adjudication;

    private static final long serialVersionUID = 1301363592L;

    /**
     * Constructor
     */
    public AddedItemDetailSubDetailComponent() {
      super();
    }

    /**
     * Constructor
     */
    public AddedItemDetailSubDetailComponent(CodeableConcept productOrService) {
      super();
      this.setProductOrService(productOrService);
    }

    /**
     * @return {@link #productOrService} (When the value is a group code then this
     *         item collects a set of related claim details, otherwise this contains
     *         the product, service, drug or other billing code for the item.)
     */
    public CodeableConcept getProductOrService() {
      if (this.productOrService == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailSubDetailComponent.productOrService");
        else if (Configuration.doAutoCreate())
          this.productOrService = new CodeableConcept(); // cc
      return this.productOrService;
    }

    public boolean hasProductOrService() {
      return this.productOrService != null && !this.productOrService.isEmpty();
    }

    /**
     * @param value {@link #productOrService} (When the value is a group code then
     *              this item collects a set of related claim details, otherwise
     *              this contains the product, service, drug or other billing code
     *              for the item.)
     */
    public AddedItemDetailSubDetailComponent setProductOrService(CodeableConcept value) {
      this.productOrService = value;
      return this;
    }

    /**
     * @return {@link #modifier} (Item typification or modifiers codes to convey
     *         additional context for the product or service.)
     */
    public List<CodeableConcept> getModifier() {
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      return this.modifier;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailSubDetailComponent setModifier(List<CodeableConcept> theModifier) {
      this.modifier = theModifier;
      return this;
    }

    public boolean hasModifier() {
      if (this.modifier == null)
        return false;
      for (CodeableConcept item : this.modifier)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public CodeableConcept addModifier() { // 3
      CodeableConcept t = new CodeableConcept();
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return t;
    }

    public AddedItemDetailSubDetailComponent addModifier(CodeableConcept t) { // 3
      if (t == null)
        return this;
      if (this.modifier == null)
        this.modifier = new ArrayList<CodeableConcept>();
      this.modifier.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #modifier}, creating
     *         it if it does not already exist {3}
     */
    public CodeableConcept getModifierFirstRep() {
      if (getModifier().isEmpty()) {
        addModifier();
      }
      return getModifier().get(0);
    }

    /**
     * @return {@link #quantity} (The number of repetitions of a service or
     *         product.)
     */
    public Quantity getQuantity() {
      if (this.quantity == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailSubDetailComponent.quantity");
        else if (Configuration.doAutoCreate())
          this.quantity = new Quantity(); // cc
      return this.quantity;
    }

    public boolean hasQuantity() {
      return this.quantity != null && !this.quantity.isEmpty();
    }

    /**
     * @param value {@link #quantity} (The number of repetitions of a service or
     *              product.)
     */
    public AddedItemDetailSubDetailComponent setQuantity(Quantity value) {
      this.quantity = value;
      return this;
    }

    /**
     * @return {@link #unitPrice} (If the item is not a group then this is the fee
     *         for the product or service, otherwise this is the total of the fees
     *         for the details of the group.)
     */
    public Money getUnitPrice() {
      if (this.unitPrice == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailSubDetailComponent.unitPrice");
        else if (Configuration.doAutoCreate())
          this.unitPrice = new Money(); // cc
      return this.unitPrice;
    }

    public boolean hasUnitPrice() {
      return this.unitPrice != null && !this.unitPrice.isEmpty();
    }

    /**
     * @param value {@link #unitPrice} (If the item is not a group then this is the
     *              fee for the product or service, otherwise this is the total of
     *              the fees for the details of the group.)
     */
    public AddedItemDetailSubDetailComponent setUnitPrice(Money value) {
      this.unitPrice = value;
      return this;
    }

    /**
     * @return {@link #factor} (A real number that represents a multiplier used in
     *         determining the overall value of services delivered and/or goods
     *         received. The concept of a Factor allows for a discount or surcharge
     *         multiplier to be applied to a monetary amount.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getFactor" gives direct access to the value
     */
    public DecimalType getFactorElement() {
      if (this.factor == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailSubDetailComponent.factor");
        else if (Configuration.doAutoCreate())
          this.factor = new DecimalType(); // bb
      return this.factor;
    }

    public boolean hasFactorElement() {
      return this.factor != null && !this.factor.isEmpty();
    }

    public boolean hasFactor() {
      return this.factor != null && !this.factor.isEmpty();
    }

    /**
     * @param value {@link #factor} (A real number that represents a multiplier used
     *              in determining the overall value of services delivered and/or
     *              goods received. The concept of a Factor allows for a discount or
     *              surcharge multiplier to be applied to a monetary amount.). This
     *              is the underlying object with id, value and extensions. The
     *              accessor "getFactor" gives direct access to the value
     */
    public AddedItemDetailSubDetailComponent setFactorElement(DecimalType value) {
      this.factor = value;
      return this;
    }

    /**
     * @return A real number that represents a multiplier used in determining the
     *         overall value of services delivered and/or goods received. The
     *         concept of a Factor allows for a discount or surcharge multiplier to
     *         be applied to a monetary amount.
     */
    public BigDecimal getFactor() {
      return this.factor == null ? null : this.factor.getValue();
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemDetailSubDetailComponent setFactor(BigDecimal value) {
      if (value == null)
        this.factor = null;
      else {
        if (this.factor == null)
          this.factor = new DecimalType();
        this.factor.setValue(value);
      }
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemDetailSubDetailComponent setFactor(long value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @param value A real number that represents a multiplier used in determining
     *              the overall value of services delivered and/or goods received.
     *              The concept of a Factor allows for a discount or surcharge
     *              multiplier to be applied to a monetary amount.
     */
    public AddedItemDetailSubDetailComponent setFactor(double value) {
      this.factor = new DecimalType();
      this.factor.setValue(value);
      return this;
    }

    /**
     * @return {@link #net} (The quantity times the unit price for an additional
     *         service or product or charge.)
     */
    public Money getNet() {
      if (this.net == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create AddedItemDetailSubDetailComponent.net");
        else if (Configuration.doAutoCreate())
          this.net = new Money(); // cc
      return this.net;
    }

    public boolean hasNet() {
      return this.net != null && !this.net.isEmpty();
    }

    /**
     * @param value {@link #net} (The quantity times the unit price for an
     *              additional service or product or charge.)
     */
    public AddedItemDetailSubDetailComponent setNet(Money value) {
      this.net = value;
      return this;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public List<PositiveIntType> getNoteNumber() {
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      return this.noteNumber;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailSubDetailComponent setNoteNumber(List<PositiveIntType> theNoteNumber) {
      this.noteNumber = theNoteNumber;
      return this;
    }

    public boolean hasNoteNumber() {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType item : this.noteNumber)
        if (!item.isEmpty())
          return true;
      return false;
    }

    /**
     * @return {@link #noteNumber} (The numbers associated with notes below which
     *         apply to the adjudication of this item.)
     */
    public PositiveIntType addNoteNumberElement() {// 2
      PositiveIntType t = new PositiveIntType();
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return t;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public AddedItemDetailSubDetailComponent addNoteNumber(int value) { // 1
      PositiveIntType t = new PositiveIntType();
      t.setValue(value);
      if (this.noteNumber == null)
        this.noteNumber = new ArrayList<PositiveIntType>();
      this.noteNumber.add(t);
      return this;
    }

    /**
     * @param value {@link #noteNumber} (The numbers associated with notes below
     *              which apply to the adjudication of this item.)
     */
    public boolean hasNoteNumber(int value) {
      if (this.noteNumber == null)
        return false;
      for (PositiveIntType v : this.noteNumber)
        if (v.getValue().equals(value)) // positiveInt
          return true;
      return false;
    }

    /**
     * @return {@link #adjudication} (The adjudication results.)
     */
    public List<AdjudicationComponent> getAdjudication() {
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      return this.adjudication;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public AddedItemDetailSubDetailComponent setAdjudication(List<AdjudicationComponent> theAdjudication) {
      this.adjudication = theAdjudication;
      return this;
    }

    public boolean hasAdjudication() {
      if (this.adjudication == null)
        return false;
      for (AdjudicationComponent item : this.adjudication)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public AdjudicationComponent addAdjudication() { // 3
      AdjudicationComponent t = new AdjudicationComponent();
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return t;
    }

    public AddedItemDetailSubDetailComponent addAdjudication(AdjudicationComponent t) { // 3
      if (t == null)
        return this;
      if (this.adjudication == null)
        this.adjudication = new ArrayList<AdjudicationComponent>();
      this.adjudication.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #adjudication},
     *         creating it if it does not already exist {3}
     */
    public AdjudicationComponent getAdjudicationFirstRep() {
      if (getAdjudication().isEmpty()) {
        addAdjudication();
      }
      return getAdjudication().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("productOrService", "CodeableConcept",
          "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
          0, 1, productOrService));
      children.add(new Property("modifier", "CodeableConcept",
          "Item typification or modifiers codes to convey additional context for the product or service.", 0,
          java.lang.Integer.MAX_VALUE, modifier));
      children.add(
          new Property("quantity", "Quantity", "The number of repetitions of a service or product.", 0, 1, quantity));
      children.add(new Property("unitPrice", "Money",
          "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
          0, 1, unitPrice));
      children.add(new Property("factor", "decimal",
          "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
          0, 1, factor));
      children.add(new Property("net", "Money",
          "The quantity times the unit price for an additional service or product or charge.", 0, 1, net));
      children.add(new Property("noteNumber", "positiveInt",
          "The numbers associated with notes below which apply to the adjudication of this item.", 0,
          java.lang.Integer.MAX_VALUE, noteNumber));
      children.add(new Property("adjudication", "@ExplanationOfBenefit.item.adjudication", "The adjudication results.",
          0, java.lang.Integer.MAX_VALUE, adjudication));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 1957227299:
        /* productOrService */ return new Property("productOrService", "CodeableConcept",
            "When the value is a group code then this item collects a set of related claim details, otherwise this contains the product, service, drug or other billing code for the item.",
            0, 1, productOrService);
      case -615513385:
        /* modifier */ return new Property("modifier", "CodeableConcept",
            "Item typification or modifiers codes to convey additional context for the product or service.", 0,
            java.lang.Integer.MAX_VALUE, modifier);
      case -1285004149:
        /* quantity */ return new Property("quantity", "Quantity", "The number of repetitions of a service or product.",
            0, 1, quantity);
      case -486196699:
        /* unitPrice */ return new Property("unitPrice", "Money",
            "If the item is not a group then this is the fee for the product or service, otherwise this is the total of the fees for the details of the group.",
            0, 1, unitPrice);
      case -1282148017:
        /* factor */ return new Property("factor", "decimal",
            "A real number that represents a multiplier used in determining the overall value of services delivered and/or goods received. The concept of a Factor allows for a discount or surcharge multiplier to be applied to a monetary amount.",
            0, 1, factor);
      case 108957:
        /* net */ return new Property("net", "Money",
            "The quantity times the unit price for an additional service or product or charge.", 0, 1, net);
      case -1110033957:
        /* noteNumber */ return new Property("noteNumber", "positiveInt",
            "The numbers associated with notes below which apply to the adjudication of this item.", 0,
            java.lang.Integer.MAX_VALUE, noteNumber);
      case -231349275:
        /* adjudication */ return new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
            "The adjudication results.", 0, java.lang.Integer.MAX_VALUE, adjudication);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 1957227299:
        /* productOrService */ return this.productOrService == null ? new Base[0]
            : new Base[] { this.productOrService }; // CodeableConcept
      case -615513385:
        /* modifier */ return this.modifier == null ? new Base[0]
            : this.modifier.toArray(new Base[this.modifier.size()]); // CodeableConcept
      case -1285004149:
        /* quantity */ return this.quantity == null ? new Base[0] : new Base[] { this.quantity }; // Quantity
      case -486196699:
        /* unitPrice */ return this.unitPrice == null ? new Base[0] : new Base[] { this.unitPrice }; // Money
      case -1282148017:
        /* factor */ return this.factor == null ? new Base[0] : new Base[] { this.factor }; // DecimalType
      case 108957:
        /* net */ return this.net == null ? new Base[0] : new Base[] { this.net }; // Money
      case -1110033957:
        /* noteNumber */ return this.noteNumber == null ? new Base[0]
            : this.noteNumber.toArray(new Base[this.noteNumber.size()]); // PositiveIntType
      case -231349275:
        /* adjudication */ return this.adjudication == null ? new Base[0]
            : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 1957227299: // productOrService
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -615513385: // modifier
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value)); // CodeableConcept
        return value;
      case -1285004149: // quantity
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
        return value;
      case -486196699: // unitPrice
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1282148017: // factor
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
        return value;
      case 108957: // net
        this.net = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1110033957: // noteNumber
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value)); // PositiveIntType
        return value;
      case -231349275: // adjudication
        this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("productOrService")) {
        this.productOrService = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("modifier")) {
        this.getModifier().add(TypeConvertor.castToCodeableConcept(value));
      } else if (name.equals("quantity")) {
        this.quantity = TypeConvertor.castToQuantity(value); // Quantity
      } else if (name.equals("unitPrice")) {
        this.unitPrice = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("factor")) {
        this.factor = TypeConvertor.castToDecimal(value); // DecimalType
      } else if (name.equals("net")) {
        this.net = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().add(TypeConvertor.castToPositiveInt(value));
      } else if (name.equals("adjudication")) {
        this.getAdjudication().add((AdjudicationComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("productOrService")) {
        this.productOrService = null;
      } else if (name.equals("modifier")) {
        this.getModifier().remove(value);
      } else if (name.equals("quantity")) {
        this.quantity = null;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = null;
      } else if (name.equals("factor")) {
        this.factor = null;
      } else if (name.equals("net")) {
        this.net = null;
      } else if (name.equals("noteNumber")) {
        this.getNoteNumber().remove(value);
      } else if (name.equals("adjudication")) {
        this.getAdjudication().remove((AdjudicationComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1957227299:
        return getProductOrService();
      case -615513385:
        return addModifier();
      case -1285004149:
        return getQuantity();
      case -486196699:
        return getUnitPrice();
      case -1282148017:
        return getFactorElement();
      case 108957:
        return getNet();
      case -1110033957:
        return addNoteNumberElement();
      case -231349275:
        return addAdjudication();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 1957227299:
        /* productOrService */ return new String[] { "CodeableConcept" };
      case -615513385:
        /* modifier */ return new String[] { "CodeableConcept" };
      case -1285004149:
        /* quantity */ return new String[] { "Quantity" };
      case -486196699:
        /* unitPrice */ return new String[] { "Money" };
      case -1282148017:
        /* factor */ return new String[] { "decimal" };
      case 108957:
        /* net */ return new String[] { "Money" };
      case -1110033957:
        /* noteNumber */ return new String[] { "positiveInt" };
      case -231349275:
        /* adjudication */ return new String[] { "@ExplanationOfBenefit.item.adjudication" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("productOrService")) {
        this.productOrService = new CodeableConcept();
        return this.productOrService;
      } else if (name.equals("modifier")) {
        return addModifier();
      } else if (name.equals("quantity")) {
        this.quantity = new Quantity();
        return this.quantity;
      } else if (name.equals("unitPrice")) {
        this.unitPrice = new Money();
        return this.unitPrice;
      } else if (name.equals("factor")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.detail.subDetail.factor");
      } else if (name.equals("net")) {
        this.net = new Money();
        return this.net;
      } else if (name.equals("noteNumber")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.addItem.detail.subDetail.noteNumber");
      } else if (name.equals("adjudication")) {
        return addAdjudication();
      } else
        return super.addChild(name);
    }

    public AddedItemDetailSubDetailComponent copy() {
      AddedItemDetailSubDetailComponent dst = new AddedItemDetailSubDetailComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(AddedItemDetailSubDetailComponent dst) {
      super.copyValues(dst);
      dst.productOrService = productOrService == null ? null : productOrService.copy();
      if (modifier != null) {
        dst.modifier = new ArrayList<CodeableConcept>();
        for (CodeableConcept i : modifier)
          dst.modifier.add(i.copy());
      }
      ;
      dst.quantity = quantity == null ? null : quantity.copy();
      dst.unitPrice = unitPrice == null ? null : unitPrice.copy();
      dst.factor = factor == null ? null : factor.copy();
      dst.net = net == null ? null : net.copy();
      if (noteNumber != null) {
        dst.noteNumber = new ArrayList<PositiveIntType>();
        for (PositiveIntType i : noteNumber)
          dst.noteNumber.add(i.copy());
      }
      ;
      if (adjudication != null) {
        dst.adjudication = new ArrayList<AdjudicationComponent>();
        for (AdjudicationComponent i : adjudication)
          dst.adjudication.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof AddedItemDetailSubDetailComponent))
        return false;
      AddedItemDetailSubDetailComponent o = (AddedItemDetailSubDetailComponent) other_;
      return compareDeep(productOrService, o.productOrService, true) && compareDeep(modifier, o.modifier, true)
          && compareDeep(quantity, o.quantity, true) && compareDeep(unitPrice, o.unitPrice, true)
          && compareDeep(factor, o.factor, true) && compareDeep(net, o.net, true)
          && compareDeep(noteNumber, o.noteNumber, true) && compareDeep(adjudication, o.adjudication, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof AddedItemDetailSubDetailComponent))
        return false;
      AddedItemDetailSubDetailComponent o = (AddedItemDetailSubDetailComponent) other_;
      return compareValues(factor, o.factor, true) && compareValues(noteNumber, o.noteNumber, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(productOrService, modifier, quantity, unitPrice,
          factor, net, noteNumber, adjudication);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.addItem.detail.subDetail";

    }

  }

  @Block()
  public static class TotalComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A code to indicate the information type of this adjudication record.
     * Information types may include: the value submitted, maximum values or
     * percentages allowed or payable under the plan, amounts that the patient is
     * responsible for in aggregate or pertaining to this item, amounts paid by
     * other coverages, and the benefit payable for this item.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 1, min = 1, max = 1, modifier = false, summary = true)
    @Description(shortDefinition = "Type of adjudication information", formalDefinition = "A code to indicate the information type of this adjudication record. Information types may include: the value submitted, maximum values or percentages allowed or payable under the plan, amounts that the patient is responsible for in aggregate or pertaining to this item, amounts paid by other coverages, and the benefit payable for this item.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/adjudication")
    protected CodeableConcept category;

    /**
     * Monetary total amount associated with the category.
     */
    @Child(name = "amount", type = { Money.class }, order = 2, min = 1, max = 1, modifier = false, summary = true)
    @Description(shortDefinition = "Financial total for the category", formalDefinition = "Monetary total amount associated with the category.")
    protected Money amount;

    private static final long serialVersionUID = 2012310309L;

    /**
     * Constructor
     */
    public TotalComponent() {
      super();
    }

    /**
     * Constructor
     */
    public TotalComponent(CodeableConcept category, Money amount) {
      super();
      this.setCategory(category);
      this.setAmount(amount);
    }

    /**
     * @return {@link #category} (A code to indicate the information type of this
     *         adjudication record. Information types may include: the value
     *         submitted, maximum values or percentages allowed or payable under the
     *         plan, amounts that the patient is responsible for in aggregate or
     *         pertaining to this item, amounts paid by other coverages, and the
     *         benefit payable for this item.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create TotalComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (A code to indicate the information type of
     *              this adjudication record. Information types may include: the
     *              value submitted, maximum values or percentages allowed or
     *              payable under the plan, amounts that the patient is responsible
     *              for in aggregate or pertaining to this item, amounts paid by
     *              other coverages, and the benefit payable for this item.)
     */
    public TotalComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #amount} (Monetary total amount associated with the category.)
     */
    public Money getAmount() {
      if (this.amount == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create TotalComponent.amount");
        else if (Configuration.doAutoCreate())
          this.amount = new Money(); // cc
      return this.amount;
    }

    public boolean hasAmount() {
      return this.amount != null && !this.amount.isEmpty();
    }

    /**
     * @param value {@link #amount} (Monetary total amount associated with the
     *              category.)
     */
    public TotalComponent setAmount(Money value) {
      this.amount = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("category", "CodeableConcept",
          "A code to indicate the information type of this adjudication record. Information types may include: the value submitted, maximum values or percentages allowed or payable under the plan, amounts that the patient is responsible for in aggregate or pertaining to this item, amounts paid by other coverages, and the benefit payable for this item.",
          0, 1, category));
      children
          .add(new Property("amount", "Money", "Monetary total amount associated with the category.", 0, 1, amount));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "A code to indicate the information type of this adjudication record. Information types may include: the value submitted, maximum values or percentages allowed or payable under the plan, amounts that the patient is responsible for in aggregate or pertaining to this item, amounts paid by other coverages, and the benefit payable for this item.",
            0, 1, category);
      case -1413853096:
        /* amount */ return new Property("amount", "Money", "Monetary total amount associated with the category.", 0, 1,
            amount);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case -1413853096:
        /* amount */ return this.amount == null ? new Base[0] : new Base[] { this.amount }; // Money
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -1413853096: // amount
        this.amount = TypeConvertor.castToMoney(value); // Money
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("amount")) {
        this.amount = TypeConvertor.castToMoney(value); // Money
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("amount")) {
        this.amount = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 50511102:
        return getCategory();
      case -1413853096:
        return getAmount();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case -1413853096:
        /* amount */ return new String[] { "Money" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("amount")) {
        this.amount = new Money();
        return this.amount;
      } else
        return super.addChild(name);
    }

    public TotalComponent copy() {
      TotalComponent dst = new TotalComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(TotalComponent dst) {
      super.copyValues(dst);
      dst.category = category == null ? null : category.copy();
      dst.amount = amount == null ? null : amount.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof TotalComponent))
        return false;
      TotalComponent o = (TotalComponent) other_;
      return compareDeep(category, o.category, true) && compareDeep(amount, o.amount, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof TotalComponent))
        return false;
      TotalComponent o = (TotalComponent) other_;
      return true;
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(category, amount);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.total";

    }

  }

  @Block()
  public static class PaymentComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Whether this represents partial or complete payment of the benefits payable.
     */
    @Child(name = "type", type = {
        CodeableConcept.class }, order = 1, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Partial or complete payment", formalDefinition = "Whether this represents partial or complete payment of the benefits payable.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-paymenttype")
    protected CodeableConcept type;

    /**
     * Total amount of all adjustments to this payment included in this transaction
     * which are not related to this claim's adjudication.
     */
    @Child(name = "adjustment", type = { Money.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Payment adjustment for non-claim issues", formalDefinition = "Total amount of all adjustments to this payment included in this transaction which are not related to this claim's adjudication.")
    protected Money adjustment;

    /**
     * Reason for the payment adjustment.
     */
    @Child(name = "adjustmentReason", type = {
        CodeableConcept.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Explanation for the variance", formalDefinition = "Reason for the payment adjustment.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/payment-adjustment-reason")
    protected CodeableConcept adjustmentReason;

    /**
     * Estimated date the payment will be issued or the actual issue date of
     * payment.
     */
    @Child(name = "date", type = { DateType.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Expected date of payment", formalDefinition = "Estimated date the payment will be issued or the actual issue date of payment.")
    protected DateType date;

    /**
     * Benefits payable less any payment adjustment.
     */
    @Child(name = "amount", type = { Money.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Payable amount after adjustment", formalDefinition = "Benefits payable less any payment adjustment.")
    protected Money amount;

    /**
     * Issuer's unique identifier for the payment instrument.
     */
    @Child(name = "identifier", type = {
        Identifier.class }, order = 6, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Business identifier for the payment", formalDefinition = "Issuer's unique identifier for the payment instrument.")
    protected Identifier identifier;

    private static final long serialVersionUID = 1539906026L;

    /**
     * Constructor
     */
    public PaymentComponent() {
      super();
    }

    /**
     * @return {@link #type} (Whether this represents partial or complete payment of
     *         the benefits payable.)
     */
    public CodeableConcept getType() {
      if (this.type == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PaymentComponent.type");
        else if (Configuration.doAutoCreate())
          this.type = new CodeableConcept(); // cc
      return this.type;
    }

    public boolean hasType() {
      return this.type != null && !this.type.isEmpty();
    }

    /**
     * @param value {@link #type} (Whether this represents partial or complete
     *              payment of the benefits payable.)
     */
    public PaymentComponent setType(CodeableConcept value) {
      this.type = value;
      return this;
    }

    /**
     * @return {@link #adjustment} (Total amount of all adjustments to this payment
     *         included in this transaction which are not related to this claim's
     *         adjudication.)
     */
    public Money getAdjustment() {
      if (this.adjustment == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PaymentComponent.adjustment");
        else if (Configuration.doAutoCreate())
          this.adjustment = new Money(); // cc
      return this.adjustment;
    }

    public boolean hasAdjustment() {
      return this.adjustment != null && !this.adjustment.isEmpty();
    }

    /**
     * @param value {@link #adjustment} (Total amount of all adjustments to this
     *              payment included in this transaction which are not related to
     *              this claim's adjudication.)
     */
    public PaymentComponent setAdjustment(Money value) {
      this.adjustment = value;
      return this;
    }

    /**
     * @return {@link #adjustmentReason} (Reason for the payment adjustment.)
     */
    public CodeableConcept getAdjustmentReason() {
      if (this.adjustmentReason == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PaymentComponent.adjustmentReason");
        else if (Configuration.doAutoCreate())
          this.adjustmentReason = new CodeableConcept(); // cc
      return this.adjustmentReason;
    }

    public boolean hasAdjustmentReason() {
      return this.adjustmentReason != null && !this.adjustmentReason.isEmpty();
    }

    /**
     * @param value {@link #adjustmentReason} (Reason for the payment adjustment.)
     */
    public PaymentComponent setAdjustmentReason(CodeableConcept value) {
      this.adjustmentReason = value;
      return this;
    }

    /**
     * @return {@link #date} (Estimated date the payment will be issued or the
     *         actual issue date of payment.). This is the underlying object with
     *         id, value and extensions. The accessor "getDate" gives direct access
     *         to the value
     */
    public DateType getDateElement() {
      if (this.date == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PaymentComponent.date");
        else if (Configuration.doAutoCreate())
          this.date = new DateType(); // bb
      return this.date;
    }

    public boolean hasDateElement() {
      return this.date != null && !this.date.isEmpty();
    }

    public boolean hasDate() {
      return this.date != null && !this.date.isEmpty();
    }

    /**
     * @param value {@link #date} (Estimated date the payment will be issued or the
     *              actual issue date of payment.). This is the underlying object
     *              with id, value and extensions. The accessor "getDate" gives
     *              direct access to the value
     */
    public PaymentComponent setDateElement(DateType value) {
      this.date = value;
      return this;
    }

    /**
     * @return Estimated date the payment will be issued or the actual issue date of
     *         payment.
     */
    public Date getDate() {
      return this.date == null ? null : this.date.getValue();
    }

    /**
     * @param value Estimated date the payment will be issued or the actual issue
     *              date of payment.
     */
    public PaymentComponent setDate(Date value) {
      if (value == null)
        this.date = null;
      else {
        if (this.date == null)
          this.date = new DateType();
        this.date.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #amount} (Benefits payable less any payment adjustment.)
     */
    public Money getAmount() {
      if (this.amount == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PaymentComponent.amount");
        else if (Configuration.doAutoCreate())
          this.amount = new Money(); // cc
      return this.amount;
    }

    public boolean hasAmount() {
      return this.amount != null && !this.amount.isEmpty();
    }

    /**
     * @param value {@link #amount} (Benefits payable less any payment adjustment.)
     */
    public PaymentComponent setAmount(Money value) {
      this.amount = value;
      return this;
    }

    /**
     * @return {@link #identifier} (Issuer's unique identifier for the payment
     *         instrument.)
     */
    public Identifier getIdentifier() {
      if (this.identifier == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create PaymentComponent.identifier");
        else if (Configuration.doAutoCreate())
          this.identifier = new Identifier(); // cc
      return this.identifier;
    }

    public boolean hasIdentifier() {
      return this.identifier != null && !this.identifier.isEmpty();
    }

    /**
     * @param value {@link #identifier} (Issuer's unique identifier for the payment
     *              instrument.)
     */
    public PaymentComponent setIdentifier(Identifier value) {
      this.identifier = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("type", "CodeableConcept",
          "Whether this represents partial or complete payment of the benefits payable.", 0, 1, type));
      children.add(new Property("adjustment", "Money",
          "Total amount of all adjustments to this payment included in this transaction which are not related to this claim's adjudication.",
          0, 1, adjustment));
      children.add(new Property("adjustmentReason", "CodeableConcept", "Reason for the payment adjustment.", 0, 1,
          adjustmentReason));
      children.add(new Property("date", "date",
          "Estimated date the payment will be issued or the actual issue date of payment.", 0, 1, date));
      children.add(new Property("amount", "Money", "Benefits payable less any payment adjustment.", 0, 1, amount));
      children.add(new Property("identifier", "Identifier", "Issuer's unique identifier for the payment instrument.", 0,
          1, identifier));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 3575610:
        /* type */ return new Property("type", "CodeableConcept",
            "Whether this represents partial or complete payment of the benefits payable.", 0, 1, type);
      case 1977085293:
        /* adjustment */ return new Property("adjustment", "Money",
            "Total amount of all adjustments to this payment included in this transaction which are not related to this claim's adjudication.",
            0, 1, adjustment);
      case -1255938543:
        /* adjustmentReason */ return new Property("adjustmentReason", "CodeableConcept",
            "Reason for the payment adjustment.", 0, 1, adjustmentReason);
      case 3076014:
        /* date */ return new Property("date", "date",
            "Estimated date the payment will be issued or the actual issue date of payment.", 0, 1, date);
      case -1413853096:
        /* amount */ return new Property("amount", "Money", "Benefits payable less any payment adjustment.", 0, 1,
            amount);
      case -1618432855:
        /* identifier */ return new Property("identifier", "Identifier",
            "Issuer's unique identifier for the payment instrument.", 0, 1, identifier);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : new Base[] { this.type }; // CodeableConcept
      case 1977085293:
        /* adjustment */ return this.adjustment == null ? new Base[0] : new Base[] { this.adjustment }; // Money
      case -1255938543:
        /* adjustmentReason */ return this.adjustmentReason == null ? new Base[0]
            : new Base[] { this.adjustmentReason }; // CodeableConcept
      case 3076014:
        /* date */ return this.date == null ? new Base[0] : new Base[] { this.date }; // DateType
      case -1413853096:
        /* amount */ return this.amount == null ? new Base[0] : new Base[] { this.amount }; // Money
      case -1618432855:
        /* identifier */ return this.identifier == null ? new Base[0] : new Base[] { this.identifier }; // Identifier
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 3575610: // type
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 1977085293: // adjustment
        this.adjustment = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1255938543: // adjustmentReason
        this.adjustmentReason = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 3076014: // date
        this.date = TypeConvertor.castToDate(value); // DateType
        return value;
      case -1413853096: // amount
        this.amount = TypeConvertor.castToMoney(value); // Money
        return value;
      case -1618432855: // identifier
        this.identifier = TypeConvertor.castToIdentifier(value); // Identifier
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("type")) {
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("adjustment")) {
        this.adjustment = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("adjustmentReason")) {
        this.adjustmentReason = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("date")) {
        this.date = TypeConvertor.castToDate(value); // DateType
      } else if (name.equals("amount")) {
        this.amount = TypeConvertor.castToMoney(value); // Money
      } else if (name.equals("identifier")) {
        this.identifier = TypeConvertor.castToIdentifier(value); // Identifier
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("type")) {
        this.type = null;
      } else if (name.equals("adjustment")) {
        this.adjustment = null;
      } else if (name.equals("adjustmentReason")) {
        this.adjustmentReason = null;
      } else if (name.equals("date")) {
        this.date = null;
      } else if (name.equals("amount")) {
        this.amount = null;
      } else if (name.equals("identifier")) {
        this.identifier = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3575610:
        return getType();
      case 1977085293:
        return getAdjustment();
      case -1255938543:
        return getAdjustmentReason();
      case 3076014:
        return getDateElement();
      case -1413853096:
        return getAmount();
      case -1618432855:
        return getIdentifier();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3575610:
        /* type */ return new String[] { "CodeableConcept" };
      case 1977085293:
        /* adjustment */ return new String[] { "Money" };
      case -1255938543:
        /* adjustmentReason */ return new String[] { "CodeableConcept" };
      case 3076014:
        /* date */ return new String[] { "date" };
      case -1413853096:
        /* amount */ return new String[] { "Money" };
      case -1618432855:
        /* identifier */ return new String[] { "Identifier" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("type")) {
        this.type = new CodeableConcept();
        return this.type;
      } else if (name.equals("adjustment")) {
        this.adjustment = new Money();
        return this.adjustment;
      } else if (name.equals("adjustmentReason")) {
        this.adjustmentReason = new CodeableConcept();
        return this.adjustmentReason;
      } else if (name.equals("date")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.payment.date");
      } else if (name.equals("amount")) {
        this.amount = new Money();
        return this.amount;
      } else if (name.equals("identifier")) {
        this.identifier = new Identifier();
        return this.identifier;
      } else
        return super.addChild(name);
    }

    public PaymentComponent copy() {
      PaymentComponent dst = new PaymentComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(PaymentComponent dst) {
      super.copyValues(dst);
      dst.type = type == null ? null : type.copy();
      dst.adjustment = adjustment == null ? null : adjustment.copy();
      dst.adjustmentReason = adjustmentReason == null ? null : adjustmentReason.copy();
      dst.date = date == null ? null : date.copy();
      dst.amount = amount == null ? null : amount.copy();
      dst.identifier = identifier == null ? null : identifier.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof PaymentComponent))
        return false;
      PaymentComponent o = (PaymentComponent) other_;
      return compareDeep(type, o.type, true) && compareDeep(adjustment, o.adjustment, true)
          && compareDeep(adjustmentReason, o.adjustmentReason, true) && compareDeep(date, o.date, true)
          && compareDeep(amount, o.amount, true) && compareDeep(identifier, o.identifier, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof PaymentComponent))
        return false;
      PaymentComponent o = (PaymentComponent) other_;
      return compareValues(date, o.date, true);
    }

    public boolean isEmpty() {
      return super.isEmpty()
          && ca.uhn.fhir.util.ElementUtil.isEmpty(type, adjustment, adjustmentReason, date, amount, identifier);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.payment";

    }

  }

  @Block()
  public static class NoteComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * A number to uniquely identify a note entry.
     */
    @Child(name = "number", type = {
        PositiveIntType.class }, order = 1, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Note instance identifier", formalDefinition = "A number to uniquely identify a note entry.")
    protected PositiveIntType number;

    /**
     * The business purpose of the note text.
     */
    @Child(name = "type", type = { CodeType.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "display | print | printoper", formalDefinition = "The business purpose of the note text.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/note-type")
    protected Enumeration<NoteType> type;

    /**
     * The explanation or description associated with the processing.
     */
    @Child(name = "text", type = { StringType.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Note explanatory text", formalDefinition = "The explanation or description associated with the processing.")
    protected StringType text;

    /**
     * A code to define the language used in the text of the note.
     */
    @Child(name = "language", type = {
        CodeableConcept.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Language of the text", formalDefinition = "A code to define the language used in the text of the note.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/languages")
    protected CodeableConcept language;

    private static final long serialVersionUID = -385184277L;

    /**
     * Constructor
     */
    public NoteComponent() {
      super();
    }

    /**
     * @return {@link #number} (A number to uniquely identify a note entry.). This
     *         is the underlying object with id, value and extensions. The accessor
     *         "getNumber" gives direct access to the value
     */
    public PositiveIntType getNumberElement() {
      if (this.number == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create NoteComponent.number");
        else if (Configuration.doAutoCreate())
          this.number = new PositiveIntType(); // bb
      return this.number;
    }

    public boolean hasNumberElement() {
      return this.number != null && !this.number.isEmpty();
    }

    public boolean hasNumber() {
      return this.number != null && !this.number.isEmpty();
    }

    /**
     * @param value {@link #number} (A number to uniquely identify a note entry.).
     *              This is the underlying object with id, value and extensions. The
     *              accessor "getNumber" gives direct access to the value
     */
    public NoteComponent setNumberElement(PositiveIntType value) {
      this.number = value;
      return this;
    }

    /**
     * @return A number to uniquely identify a note entry.
     */
    public int getNumber() {
      return this.number == null || this.number.isEmpty() ? 0 : this.number.getValue();
    }

    /**
     * @param value A number to uniquely identify a note entry.
     */
    public NoteComponent setNumber(int value) {
      if (this.number == null)
        this.number = new PositiveIntType();
      this.number.setValue(value);
      return this;
    }

    /**
     * @return {@link #type} (The business purpose of the note text.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getType" gives direct access to the value
     */
    public Enumeration<NoteType> getTypeElement() {
      if (this.type == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create NoteComponent.type");
        else if (Configuration.doAutoCreate())
          this.type = new Enumeration<NoteType>(new NoteTypeEnumFactory()); // bb
      return this.type;
    }

    public boolean hasTypeElement() {
      return this.type != null && !this.type.isEmpty();
    }

    public boolean hasType() {
      return this.type != null && !this.type.isEmpty();
    }

    /**
     * @param value {@link #type} (The business purpose of the note text.). This is
     *              the underlying object with id, value and extensions. The
     *              accessor "getType" gives direct access to the value
     */
    public NoteComponent setTypeElement(Enumeration<NoteType> value) {
      this.type = value;
      return this;
    }

    /**
     * @return The business purpose of the note text.
     */
    public NoteType getType() {
      return this.type == null ? null : this.type.getValue();
    }

    /**
     * @param value The business purpose of the note text.
     */
    public NoteComponent setType(NoteType value) {
      if (value == null)
        this.type = null;
      else {
        if (this.type == null)
          this.type = new Enumeration<NoteType>(new NoteTypeEnumFactory());
        this.type.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #text} (The explanation or description associated with the
     *         processing.). This is the underlying object with id, value and
     *         extensions. The accessor "getText" gives direct access to the value
     */
    public StringType getTextElement() {
      if (this.text == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create NoteComponent.text");
        else if (Configuration.doAutoCreate())
          this.text = new StringType(); // bb
      return this.text;
    }

    public boolean hasTextElement() {
      return this.text != null && !this.text.isEmpty();
    }

    public boolean hasText() {
      return this.text != null && !this.text.isEmpty();
    }

    /**
     * @param value {@link #text} (The explanation or description associated with
     *              the processing.). This is the underlying object with id, value
     *              and extensions. The accessor "getText" gives direct access to
     *              the value
     */
    public NoteComponent setTextElement(StringType value) {
      this.text = value;
      return this;
    }

    /**
     * @return The explanation or description associated with the processing.
     */
    public String getText() {
      return this.text == null ? null : this.text.getValue();
    }

    /**
     * @param value The explanation or description associated with the processing.
     */
    public NoteComponent setText(String value) {
      if (Utilities.noString(value))
        this.text = null;
      else {
        if (this.text == null)
          this.text = new StringType();
        this.text.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #language} (A code to define the language used in the text of
     *         the note.)
     */
    public CodeableConcept getLanguage() {
      if (this.language == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create NoteComponent.language");
        else if (Configuration.doAutoCreate())
          this.language = new CodeableConcept(); // cc
      return this.language;
    }

    public boolean hasLanguage() {
      return this.language != null && !this.language.isEmpty();
    }

    /**
     * @param value {@link #language} (A code to define the language used in the
     *              text of the note.)
     */
    public NoteComponent setLanguage(CodeableConcept value) {
      this.language = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("number", "positiveInt", "A number to uniquely identify a note entry.", 0, 1, number));
      children.add(new Property("type", "code", "The business purpose of the note text.", 0, 1, type));
      children.add(
          new Property("text", "string", "The explanation or description associated with the processing.", 0, 1, text));
      children.add(new Property("language", "CodeableConcept",
          "A code to define the language used in the text of the note.", 0, 1, language));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case -1034364087:
        /* number */ return new Property("number", "positiveInt", "A number to uniquely identify a note entry.", 0, 1,
            number);
      case 3575610:
        /* type */ return new Property("type", "code", "The business purpose of the note text.", 0, 1, type);
      case 3556653:
        /* text */ return new Property("text", "string",
            "The explanation or description associated with the processing.", 0, 1, text);
      case -1613589672:
        /* language */ return new Property("language", "CodeableConcept",
            "A code to define the language used in the text of the note.", 0, 1, language);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case -1034364087:
        /* number */ return this.number == null ? new Base[0] : new Base[] { this.number }; // PositiveIntType
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : new Base[] { this.type }; // Enumeration<NoteType>
      case 3556653:
        /* text */ return this.text == null ? new Base[0] : new Base[] { this.text }; // StringType
      case -1613589672:
        /* language */ return this.language == null ? new Base[0] : new Base[] { this.language }; // CodeableConcept
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case -1034364087: // number
        this.number = TypeConvertor.castToPositiveInt(value); // PositiveIntType
        return value;
      case 3575610: // type
        value = new NoteTypeEnumFactory().fromType(TypeConvertor.castToCode(value));
        this.type = (Enumeration) value; // Enumeration<NoteType>
        return value;
      case 3556653: // text
        this.text = TypeConvertor.castToString(value); // StringType
        return value;
      case -1613589672: // language
        this.language = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("number")) {
        this.number = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      } else if (name.equals("type")) {
        value = new NoteTypeEnumFactory().fromType(TypeConvertor.castToCode(value));
        this.type = (Enumeration) value; // Enumeration<NoteType>
      } else if (name.equals("text")) {
        this.text = TypeConvertor.castToString(value); // StringType
      } else if (name.equals("language")) {
        this.language = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("number")) {
        this.number = null;
      } else if (name.equals("type")) {
        this.type = null;
      } else if (name.equals("text")) {
        this.text = null;
      } else if (name.equals("language")) {
        this.language = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case -1034364087:
        return getNumberElement();
      case 3575610:
        return getTypeElement();
      case 3556653:
        return getTextElement();
      case -1613589672:
        return getLanguage();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case -1034364087:
        /* number */ return new String[] { "positiveInt" };
      case 3575610:
        /* type */ return new String[] { "code" };
      case 3556653:
        /* text */ return new String[] { "string" };
      case -1613589672:
        /* language */ return new String[] { "CodeableConcept" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("number")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.processNote.number");
      } else if (name.equals("type")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.processNote.type");
      } else if (name.equals("text")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.processNote.text");
      } else if (name.equals("language")) {
        this.language = new CodeableConcept();
        return this.language;
      } else
        return super.addChild(name);
    }

    public NoteComponent copy() {
      NoteComponent dst = new NoteComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(NoteComponent dst) {
      super.copyValues(dst);
      dst.number = number == null ? null : number.copy();
      dst.type = type == null ? null : type.copy();
      dst.text = text == null ? null : text.copy();
      dst.language = language == null ? null : language.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof NoteComponent))
        return false;
      NoteComponent o = (NoteComponent) other_;
      return compareDeep(number, o.number, true) && compareDeep(type, o.type, true) && compareDeep(text, o.text, true)
          && compareDeep(language, o.language, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof NoteComponent))
        return false;
      NoteComponent o = (NoteComponent) other_;
      return compareValues(number, o.number, true) && compareValues(type, o.type, true)
          && compareValues(text, o.text, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(number, type, text, language);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.processNote";

    }

  }

  @Block()
  public static class BenefitBalanceComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Code to identify the general type of benefits under which products and
     * services are provided.
     */
    @Child(name = "category", type = {
        CodeableConcept.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefit classification", formalDefinition = "Code to identify the general type of benefits under which products and services are provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/ex-benefitcategory")
    protected CodeableConcept category;

    /**
     * True if the indicated class of service is excluded from the plan, missing or
     * False indicates the product or service is included in the coverage.
     */
    @Child(name = "excluded", type = {
        BooleanType.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Excluded from the plan", formalDefinition = "True if the indicated class of service is excluded from the plan, missing or False indicates the product or service is included in the coverage.")
    protected BooleanType excluded;

    /**
     * A short name or tag for the benefit.
     */
    @Child(name = "name", type = { StringType.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Short name for the benefit", formalDefinition = "A short name or tag for the benefit.")
    protected StringType name;

    /**
     * A richer description of the benefit or services covered.
     */
    @Child(name = "description", type = {
        StringType.class }, order = 4, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Description of the benefit or services covered", formalDefinition = "A richer description of the benefit or services covered.")
    protected StringType description;

    /**
     * Is a flag to indicate whether the benefits refer to in-network providers or
     * out-of-network providers.
     */
    @Child(name = "network", type = {
        CodeableConcept.class }, order = 5, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "In or out of network", formalDefinition = "Is a flag to indicate whether the benefits refer to in-network providers or out-of-network providers.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/benefit-network")
    protected CodeableConcept network;

    /**
     * Indicates if the benefits apply to an individual or to the family.
     */
    @Child(name = "unit", type = {
        CodeableConcept.class }, order = 6, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Individual or family", formalDefinition = "Indicates if the benefits apply to an individual or to the family.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/benefit-unit")
    protected CodeableConcept unit;

    /**
     * The term or period of the values such as 'maximum lifetime benefit' or
     * 'maximum annual visits'.
     */
    @Child(name = "term", type = {
        CodeableConcept.class }, order = 7, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Annual or lifetime", formalDefinition = "The term or period of the values such as 'maximum lifetime benefit' or 'maximum annual visits'.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/benefit-term")
    protected CodeableConcept term;

    /**
     * Benefits Used to date.
     */
    @Child(name = "financial", type = {}, order = 8, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
    @Description(shortDefinition = "Benefit Summary", formalDefinition = "Benefits Used to date.")
    protected List<BenefitComponent> financial;

    private static final long serialVersionUID = -1889655824L;

    /**
     * Constructor
     */
    public BenefitBalanceComponent() {
      super();
    }

    /**
     * Constructor
     */
    public BenefitBalanceComponent(CodeableConcept category) {
      super();
      this.setCategory(category);
    }

    /**
     * @return {@link #category} (Code to identify the general type of benefits
     *         under which products and services are provided.)
     */
    public CodeableConcept getCategory() {
      if (this.category == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.category");
        else if (Configuration.doAutoCreate())
          this.category = new CodeableConcept(); // cc
      return this.category;
    }

    public boolean hasCategory() {
      return this.category != null && !this.category.isEmpty();
    }

    /**
     * @param value {@link #category} (Code to identify the general type of benefits
     *              under which products and services are provided.)
     */
    public BenefitBalanceComponent setCategory(CodeableConcept value) {
      this.category = value;
      return this;
    }

    /**
     * @return {@link #excluded} (True if the indicated class of service is excluded
     *         from the plan, missing or False indicates the product or service is
     *         included in the coverage.). This is the underlying object with id,
     *         value and extensions. The accessor "getExcluded" gives direct access
     *         to the value
     */
    public BooleanType getExcludedElement() {
      if (this.excluded == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.excluded");
        else if (Configuration.doAutoCreate())
          this.excluded = new BooleanType(); // bb
      return this.excluded;
    }

    public boolean hasExcludedElement() {
      return this.excluded != null && !this.excluded.isEmpty();
    }

    public boolean hasExcluded() {
      return this.excluded != null && !this.excluded.isEmpty();
    }

    /**
     * @param value {@link #excluded} (True if the indicated class of service is
     *              excluded from the plan, missing or False indicates the product
     *              or service is included in the coverage.). This is the underlying
     *              object with id, value and extensions. The accessor "getExcluded"
     *              gives direct access to the value
     */
    public BenefitBalanceComponent setExcludedElement(BooleanType value) {
      this.excluded = value;
      return this;
    }

    /**
     * @return True if the indicated class of service is excluded from the plan,
     *         missing or False indicates the product or service is included in the
     *         coverage.
     */
    public boolean getExcluded() {
      return this.excluded == null || this.excluded.isEmpty() ? false : this.excluded.getValue();
    }

    /**
     * @param value True if the indicated class of service is excluded from the
     *              plan, missing or False indicates the product or service is
     *              included in the coverage.
     */
    public BenefitBalanceComponent setExcluded(boolean value) {
      if (this.excluded == null)
        this.excluded = new BooleanType();
      this.excluded.setValue(value);
      return this;
    }

    /**
     * @return {@link #name} (A short name or tag for the benefit.). This is the
     *         underlying object with id, value and extensions. The accessor
     *         "getName" gives direct access to the value
     */
    public StringType getNameElement() {
      if (this.name == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.name");
        else if (Configuration.doAutoCreate())
          this.name = new StringType(); // bb
      return this.name;
    }

    public boolean hasNameElement() {
      return this.name != null && !this.name.isEmpty();
    }

    public boolean hasName() {
      return this.name != null && !this.name.isEmpty();
    }

    /**
     * @param value {@link #name} (A short name or tag for the benefit.). This is
     *              the underlying object with id, value and extensions. The
     *              accessor "getName" gives direct access to the value
     */
    public BenefitBalanceComponent setNameElement(StringType value) {
      this.name = value;
      return this;
    }

    /**
     * @return A short name or tag for the benefit.
     */
    public String getName() {
      return this.name == null ? null : this.name.getValue();
    }

    /**
     * @param value A short name or tag for the benefit.
     */
    public BenefitBalanceComponent setName(String value) {
      if (Utilities.noString(value))
        this.name = null;
      else {
        if (this.name == null)
          this.name = new StringType();
        this.name.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #description} (A richer description of the benefit or services
     *         covered.). This is the underlying object with id, value and
     *         extensions. The accessor "getDescription" gives direct access to the
     *         value
     */
    public StringType getDescriptionElement() {
      if (this.description == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.description");
        else if (Configuration.doAutoCreate())
          this.description = new StringType(); // bb
      return this.description;
    }

    public boolean hasDescriptionElement() {
      return this.description != null && !this.description.isEmpty();
    }

    public boolean hasDescription() {
      return this.description != null && !this.description.isEmpty();
    }

    /**
     * @param value {@link #description} (A richer description of the benefit or
     *              services covered.). This is the underlying object with id, value
     *              and extensions. The accessor "getDescription" gives direct
     *              access to the value
     */
    public BenefitBalanceComponent setDescriptionElement(StringType value) {
      this.description = value;
      return this;
    }

    /**
     * @return A richer description of the benefit or services covered.
     */
    public String getDescription() {
      return this.description == null ? null : this.description.getValue();
    }

    /**
     * @param value A richer description of the benefit or services covered.
     */
    public BenefitBalanceComponent setDescription(String value) {
      if (Utilities.noString(value))
        this.description = null;
      else {
        if (this.description == null)
          this.description = new StringType();
        this.description.setValue(value);
      }
      return this;
    }

    /**
     * @return {@link #network} (Is a flag to indicate whether the benefits refer to
     *         in-network providers or out-of-network providers.)
     */
    public CodeableConcept getNetwork() {
      if (this.network == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.network");
        else if (Configuration.doAutoCreate())
          this.network = new CodeableConcept(); // cc
      return this.network;
    }

    public boolean hasNetwork() {
      return this.network != null && !this.network.isEmpty();
    }

    /**
     * @param value {@link #network} (Is a flag to indicate whether the benefits
     *              refer to in-network providers or out-of-network providers.)
     */
    public BenefitBalanceComponent setNetwork(CodeableConcept value) {
      this.network = value;
      return this;
    }

    /**
     * @return {@link #unit} (Indicates if the benefits apply to an individual or to
     *         the family.)
     */
    public CodeableConcept getUnit() {
      if (this.unit == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.unit");
        else if (Configuration.doAutoCreate())
          this.unit = new CodeableConcept(); // cc
      return this.unit;
    }

    public boolean hasUnit() {
      return this.unit != null && !this.unit.isEmpty();
    }

    /**
     * @param value {@link #unit} (Indicates if the benefits apply to an individual
     *              or to the family.)
     */
    public BenefitBalanceComponent setUnit(CodeableConcept value) {
      this.unit = value;
      return this;
    }

    /**
     * @return {@link #term} (The term or period of the values such as 'maximum
     *         lifetime benefit' or 'maximum annual visits'.)
     */
    public CodeableConcept getTerm() {
      if (this.term == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitBalanceComponent.term");
        else if (Configuration.doAutoCreate())
          this.term = new CodeableConcept(); // cc
      return this.term;
    }

    public boolean hasTerm() {
      return this.term != null && !this.term.isEmpty();
    }

    /**
     * @param value {@link #term} (The term or period of the values such as 'maximum
     *              lifetime benefit' or 'maximum annual visits'.)
     */
    public BenefitBalanceComponent setTerm(CodeableConcept value) {
      this.term = value;
      return this;
    }

    /**
     * @return {@link #financial} (Benefits Used to date.)
     */
    public List<BenefitComponent> getFinancial() {
      if (this.financial == null)
        this.financial = new ArrayList<BenefitComponent>();
      return this.financial;
    }

    /**
     * @return Returns a reference to <code>this</code> for easy method chaining
     */
    public BenefitBalanceComponent setFinancial(List<BenefitComponent> theFinancial) {
      this.financial = theFinancial;
      return this;
    }

    public boolean hasFinancial() {
      if (this.financial == null)
        return false;
      for (BenefitComponent item : this.financial)
        if (!item.isEmpty())
          return true;
      return false;
    }

    public BenefitComponent addFinancial() { // 3
      BenefitComponent t = new BenefitComponent();
      if (this.financial == null)
        this.financial = new ArrayList<BenefitComponent>();
      this.financial.add(t);
      return t;
    }

    public BenefitBalanceComponent addFinancial(BenefitComponent t) { // 3
      if (t == null)
        return this;
      if (this.financial == null)
        this.financial = new ArrayList<BenefitComponent>();
      this.financial.add(t);
      return this;
    }

    /**
     * @return The first repetition of repeating field {@link #financial}, creating
     *         it if it does not already exist {3}
     */
    public BenefitComponent getFinancialFirstRep() {
      if (getFinancial().isEmpty()) {
        addFinancial();
      }
      return getFinancial().get(0);
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("category", "CodeableConcept",
          "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
          category));
      children.add(new Property("excluded", "boolean",
          "True if the indicated class of service is excluded from the plan, missing or False indicates the product or service is included in the coverage.",
          0, 1, excluded));
      children.add(new Property("name", "string", "A short name or tag for the benefit.", 0, 1, name));
      children.add(new Property("description", "string", "A richer description of the benefit or services covered.", 0,
          1, description));
      children.add(new Property("network", "CodeableConcept",
          "Is a flag to indicate whether the benefits refer to in-network providers or out-of-network providers.", 0, 1,
          network));
      children.add(new Property("unit", "CodeableConcept",
          "Indicates if the benefits apply to an individual or to the family.", 0, 1, unit));
      children.add(new Property("term", "CodeableConcept",
          "The term or period of the values such as 'maximum lifetime benefit' or 'maximum annual visits'.", 0, 1,
          term));
      children.add(new Property("financial", "", "Benefits Used to date.", 0, java.lang.Integer.MAX_VALUE, financial));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 50511102:
        /* category */ return new Property("category", "CodeableConcept",
            "Code to identify the general type of benefits under which products and services are provided.", 0, 1,
            category);
      case 1994055114:
        /* excluded */ return new Property("excluded", "boolean",
            "True if the indicated class of service is excluded from the plan, missing or False indicates the product or service is included in the coverage.",
            0, 1, excluded);
      case 3373707:
        /* name */ return new Property("name", "string", "A short name or tag for the benefit.", 0, 1, name);
      case -1724546052:
        /* description */ return new Property("description", "string",
            "A richer description of the benefit or services covered.", 0, 1, description);
      case 1843485230:
        /* network */ return new Property("network", "CodeableConcept",
            "Is a flag to indicate whether the benefits refer to in-network providers or out-of-network providers.", 0,
            1, network);
      case 3594628:
        /* unit */ return new Property("unit", "CodeableConcept",
            "Indicates if the benefits apply to an individual or to the family.", 0, 1, unit);
      case 3556460:
        /* term */ return new Property("term", "CodeableConcept",
            "The term or period of the values such as 'maximum lifetime benefit' or 'maximum annual visits'.", 0, 1,
            term);
      case 357555337:
        /* financial */ return new Property("financial", "", "Benefits Used to date.", 0, java.lang.Integer.MAX_VALUE,
            financial);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 50511102:
        /* category */ return this.category == null ? new Base[0] : new Base[] { this.category }; // CodeableConcept
      case 1994055114:
        /* excluded */ return this.excluded == null ? new Base[0] : new Base[] { this.excluded }; // BooleanType
      case 3373707:
        /* name */ return this.name == null ? new Base[0] : new Base[] { this.name }; // StringType
      case -1724546052:
        /* description */ return this.description == null ? new Base[0] : new Base[] { this.description }; // StringType
      case 1843485230:
        /* network */ return this.network == null ? new Base[0] : new Base[] { this.network }; // CodeableConcept
      case 3594628:
        /* unit */ return this.unit == null ? new Base[0] : new Base[] { this.unit }; // CodeableConcept
      case 3556460:
        /* term */ return this.term == null ? new Base[0] : new Base[] { this.term }; // CodeableConcept
      case 357555337:
        /* financial */ return this.financial == null ? new Base[0]
            : this.financial.toArray(new Base[this.financial.size()]); // BenefitComponent
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 50511102: // category
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 1994055114: // excluded
        this.excluded = TypeConvertor.castToBoolean(value); // BooleanType
        return value;
      case 3373707: // name
        this.name = TypeConvertor.castToString(value); // StringType
        return value;
      case -1724546052: // description
        this.description = TypeConvertor.castToString(value); // StringType
        return value;
      case 1843485230: // network
        this.network = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 3594628: // unit
        this.unit = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 3556460: // term
        this.term = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case 357555337: // financial
        this.getFinancial().add((BenefitComponent) value); // BenefitComponent
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("category")) {
        this.category = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("excluded")) {
        this.excluded = TypeConvertor.castToBoolean(value); // BooleanType
      } else if (name.equals("name")) {
        this.name = TypeConvertor.castToString(value); // StringType
      } else if (name.equals("description")) {
        this.description = TypeConvertor.castToString(value); // StringType
      } else if (name.equals("network")) {
        this.network = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("unit")) {
        this.unit = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("term")) {
        this.term = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("financial")) {
        this.getFinancial().add((BenefitComponent) value);
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("category")) {
        this.category = null;
      } else if (name.equals("excluded")) {
        this.excluded = null;
      } else if (name.equals("name")) {
        this.name = null;
      } else if (name.equals("description")) {
        this.description = null;
      } else if (name.equals("network")) {
        this.network = null;
      } else if (name.equals("unit")) {
        this.unit = null;
      } else if (name.equals("term")) {
        this.term = null;
      } else if (name.equals("financial")) {
        this.getFinancial().remove((BenefitComponent) value);
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 50511102:
        return getCategory();
      case 1994055114:
        return getExcludedElement();
      case 3373707:
        return getNameElement();
      case -1724546052:
        return getDescriptionElement();
      case 1843485230:
        return getNetwork();
      case 3594628:
        return getUnit();
      case 3556460:
        return getTerm();
      case 357555337:
        return addFinancial();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 50511102:
        /* category */ return new String[] { "CodeableConcept" };
      case 1994055114:
        /* excluded */ return new String[] { "boolean" };
      case 3373707:
        /* name */ return new String[] { "string" };
      case -1724546052:
        /* description */ return new String[] { "string" };
      case 1843485230:
        /* network */ return new String[] { "CodeableConcept" };
      case 3594628:
        /* unit */ return new String[] { "CodeableConcept" };
      case 3556460:
        /* term */ return new String[] { "CodeableConcept" };
      case 357555337:
        /* financial */ return new String[] {};
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("category")) {
        this.category = new CodeableConcept();
        return this.category;
      } else if (name.equals("excluded")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.benefitBalance.excluded");
      } else if (name.equals("name")) {
        throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.benefitBalance.name");
      } else if (name.equals("description")) {
        throw new FHIRException(
            "Cannot call addChild on a singleton property ExplanationOfBenefit.benefitBalance.description");
      } else if (name.equals("network")) {
        this.network = new CodeableConcept();
        return this.network;
      } else if (name.equals("unit")) {
        this.unit = new CodeableConcept();
        return this.unit;
      } else if (name.equals("term")) {
        this.term = new CodeableConcept();
        return this.term;
      } else if (name.equals("financial")) {
        return addFinancial();
      } else
        return super.addChild(name);
    }

    public BenefitBalanceComponent copy() {
      BenefitBalanceComponent dst = new BenefitBalanceComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(BenefitBalanceComponent dst) {
      super.copyValues(dst);
      dst.category = category == null ? null : category.copy();
      dst.excluded = excluded == null ? null : excluded.copy();
      dst.name = name == null ? null : name.copy();
      dst.description = description == null ? null : description.copy();
      dst.network = network == null ? null : network.copy();
      dst.unit = unit == null ? null : unit.copy();
      dst.term = term == null ? null : term.copy();
      if (financial != null) {
        dst.financial = new ArrayList<BenefitComponent>();
        for (BenefitComponent i : financial)
          dst.financial.add(i.copy());
      }
      ;
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof BenefitBalanceComponent))
        return false;
      BenefitBalanceComponent o = (BenefitBalanceComponent) other_;
      return compareDeep(category, o.category, true) && compareDeep(excluded, o.excluded, true)
          && compareDeep(name, o.name, true) && compareDeep(description, o.description, true)
          && compareDeep(network, o.network, true) && compareDeep(unit, o.unit, true) && compareDeep(term, o.term, true)
          && compareDeep(financial, o.financial, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof BenefitBalanceComponent))
        return false;
      BenefitBalanceComponent o = (BenefitBalanceComponent) other_;
      return compareValues(excluded, o.excluded, true) && compareValues(name, o.name, true)
          && compareValues(description, o.description, true);
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(category, excluded, name, description, network,
          unit, term, financial);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.benefitBalance";

    }

  }

  @Block()
  public static class BenefitComponent extends BackboneElement implements IBaseBackboneElement {
    /**
     * Classification of benefit being provided.
     */
    @Child(name = "type", type = {
        CodeableConcept.class }, order = 1, min = 1, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefit classification", formalDefinition = "Classification of benefit being provided.")
    @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/benefit-type")
    protected CodeableConcept type;

    /**
     * The quantity of the benefit which is permitted under the coverage.
     */
    @Child(name = "allowed", type = { UnsignedIntType.class, StringType.class,
        Money.class }, order = 2, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefits allowed", formalDefinition = "The quantity of the benefit which is permitted under the coverage.")
    protected DataType allowed;

    /**
     * The quantity of the benefit which have been consumed to date.
     */
    @Child(name = "used", type = { UnsignedIntType.class,
        Money.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
    @Description(shortDefinition = "Benefits used", formalDefinition = "The quantity of the benefit which have been consumed to date.")
    protected DataType used;

    private static final long serialVersionUID = 1900247614L;

    /**
     * Constructor
     */
    public BenefitComponent() {
      super();
    }

    /**
     * Constructor
     */
    public BenefitComponent(CodeableConcept type) {
      super();
      this.setType(type);
    }

    /**
     * @return {@link #type} (Classification of benefit being provided.)
     */
    public CodeableConcept getType() {
      if (this.type == null)
        if (Configuration.errorOnAutoCreate())
          throw new Error("Attempt to auto-create BenefitComponent.type");
        else if (Configuration.doAutoCreate())
          this.type = new CodeableConcept(); // cc
      return this.type;
    }

    public boolean hasType() {
      return this.type != null && !this.type.isEmpty();
    }

    /**
     * @param value {@link #type} (Classification of benefit being provided.)
     */
    public BenefitComponent setType(CodeableConcept value) {
      this.type = value;
      return this;
    }

    /**
     * @return {@link #allowed} (The quantity of the benefit which is permitted
     *         under the coverage.)
     */
    public DataType getAllowed() {
      return this.allowed;
    }

    /**
     * @return {@link #allowed} (The quantity of the benefit which is permitted
     *         under the coverage.)
     */
    public UnsignedIntType getAllowedUnsignedIntType() throws FHIRException {
      if (this.allowed == null)
        this.allowed = new UnsignedIntType();
      if (!(this.allowed instanceof UnsignedIntType))
        throw new FHIRException("Type mismatch: the type UnsignedIntType was expected, but "
            + this.allowed.getClass().getName() + " was encountered");
      return (UnsignedIntType) this.allowed;
    }

    public boolean hasAllowedUnsignedIntType() {
        return this.allowed instanceof UnsignedIntType;
    }

    /**
     * @return {@link #allowed} (The quantity of the benefit which is permitted
     *         under the coverage.)
     */
    public StringType getAllowedStringType() throws FHIRException {
      if (this.allowed == null)
        this.allowed = new StringType();
      if (!(this.allowed instanceof StringType))
        throw new FHIRException("Type mismatch: the type StringType was expected, but "
            + this.allowed.getClass().getName() + " was encountered");
      return (StringType) this.allowed;
    }

    public boolean hasAllowedStringType() {
        return this.allowed instanceof StringType;
    }

    /**
     * @return {@link #allowed} (The quantity of the benefit which is permitted
     *         under the coverage.)
     */
    public Money getAllowedMoney() throws FHIRException {
      if (this.allowed == null)
        this.allowed = new Money();
      if (!(this.allowed instanceof Money))
        throw new FHIRException("Type mismatch: the type Money was expected, but " + this.allowed.getClass().getName()
            + " was encountered");
      return (Money) this.allowed;
    }

    public boolean hasAllowedMoney() {
        return this.allowed instanceof Money;
    }

    public boolean hasAllowed() {
      return this.allowed != null && !this.allowed.isEmpty();
    }

    /**
     * @param value {@link #allowed} (The quantity of the benefit which is permitted
     *              under the coverage.)
     */
    public BenefitComponent setAllowed(DataType value) {
      if (value != null && !(value instanceof UnsignedIntType || value instanceof StringType || value instanceof Money))
        throw new Error(
            "Not the right type for ExplanationOfBenefit.benefitBalance.financial.allowed[x]: " + value.fhirType());
      this.allowed = value;
      return this;
    }

    /**
     * @return {@link #used} (The quantity of the benefit which have been consumed
     *         to date.)
     */
    public DataType getUsed() {
      return this.used;
    }

    /**
     * @return {@link #used} (The quantity of the benefit which have been consumed
     *         to date.)
     */
    public UnsignedIntType getUsedUnsignedIntType() throws FHIRException {
      if (this.used == null)
        this.used = new UnsignedIntType();
      if (!(this.used instanceof UnsignedIntType))
        throw new FHIRException("Type mismatch: the type UnsignedIntType was expected, but "
            + this.used.getClass().getName() + " was encountered");
      return (UnsignedIntType) this.used;
    }

    public boolean hasUsedUnsignedIntType() {
        return this.used instanceof UnsignedIntType;
    }

    /**
     * @return {@link #used} (The quantity of the benefit which have been consumed
     *         to date.)
     */
    public Money getUsedMoney() throws FHIRException {
      if (this.used == null)
        this.used = new Money();
      if (!(this.used instanceof Money))
        throw new FHIRException(
            "Type mismatch: the type Money was expected, but " + this.used.getClass().getName() + " was encountered");
      return (Money) this.used;
    }

    public boolean hasUsedMoney() {
        return this.used instanceof Money;
    }

    public boolean hasUsed() {
      return this.used != null && !this.used.isEmpty();
    }

    /**
     * @param value {@link #used} (The quantity of the benefit which have been
     *              consumed to date.)
     */
    public BenefitComponent setUsed(DataType value) {
      if (value != null && !(value instanceof UnsignedIntType || value instanceof Money))
        throw new Error(
            "Not the right type for ExplanationOfBenefit.benefitBalance.financial.used[x]: " + value.fhirType());
      this.used = value;
      return this;
    }

    protected void listChildren(List<Property> children) {
      super.listChildren(children);
      children.add(new Property("type", "CodeableConcept", "Classification of benefit being provided.", 0, 1, type));
      children.add(new Property("allowed[x]", "unsignedInt|string|Money",
          "The quantity of the benefit which is permitted under the coverage.", 0, 1, allowed));
      children.add(new Property("used[x]", "unsignedInt|Money",
          "The quantity of the benefit which have been consumed to date.", 0, 1, used));
    }

    @Override
    public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
      switch (_hash) {
      case 3575610:
        /* type */ return new Property("type", "CodeableConcept", "Classification of benefit being provided.", 0, 1,
            type);
      case -1336663592:
        /* allowed[x] */ return new Property("allowed[x]", "unsignedInt|string|Money",
            "The quantity of the benefit which is permitted under the coverage.", 0, 1, allowed);
      case -911343192:
        /* allowed */ return new Property("allowed[x]", "unsignedInt|string|Money",
            "The quantity of the benefit which is permitted under the coverage.", 0, 1, allowed);
      case 1668802034:
        /* allowedUnsignedInt */ return new Property("allowed[x]", "unsignedInt",
            "The quantity of the benefit which is permitted under the coverage.", 0, 1, allowed);
      case -2135265319:
        /* allowedString */ return new Property("allowed[x]", "string",
            "The quantity of the benefit which is permitted under the coverage.", 0, 1, allowed);
      case -351668232:
        /* allowedMoney */ return new Property("allowed[x]", "Money",
            "The quantity of the benefit which is permitted under the coverage.", 0, 1, allowed);
      case -147553373:
        /* used[x] */ return new Property("used[x]", "unsignedInt|Money",
            "The quantity of the benefit which have been consumed to date.", 0, 1, used);
      case 3599293:
        /* used */ return new Property("used[x]", "unsignedInt|Money",
            "The quantity of the benefit which have been consumed to date.", 0, 1, used);
      case 1252740285:
        /* usedUnsignedInt */ return new Property("used[x]", "unsignedInt",
            "The quantity of the benefit which have been consumed to date.", 0, 1, used);
      case -78048509:
        /* usedMoney */ return new Property("used[x]", "Money",
            "The quantity of the benefit which have been consumed to date.", 0, 1, used);
      default:
        return super.getNamedProperty(_hash, _name, _checkValid);
      }

    }

    @Override
    public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
      switch (hash) {
      case 3575610:
        /* type */ return this.type == null ? new Base[0] : new Base[] { this.type }; // CodeableConcept
      case -911343192:
        /* allowed */ return this.allowed == null ? new Base[0] : new Base[] { this.allowed }; // DataType
      case 3599293:
        /* used */ return this.used == null ? new Base[0] : new Base[] { this.used }; // DataType
      default:
        return super.getProperty(hash, name, checkValid);
      }

    }

    @Override
    public Base setProperty(int hash, String name, Base value) throws FHIRException {
      switch (hash) {
      case 3575610: // type
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
        return value;
      case -911343192: // allowed
        this.allowed = TypeConvertor.castToType(value); // DataType
        return value;
      case 3599293: // used
        this.used = TypeConvertor.castToType(value); // DataType
        return value;
      default:
        return super.setProperty(hash, name, value);
      }

    }

    @Override
    public Base setProperty(String name, Base value) throws FHIRException {
      if (name.equals("type")) {
        this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      } else if (name.equals("allowed[x]")) {
        this.allowed = TypeConvertor.castToType(value); // DataType
      } else if (name.equals("used[x]")) {
        this.used = TypeConvertor.castToType(value); // DataType
      } else
        return super.setProperty(name, value);
      return value;
    }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
      if (name.equals("type")) {
        this.type = null;
      } else if (name.equals("allowed[x]")) {
        this.allowed = null;
      } else if (name.equals("used[x]")) {
        this.used = null;
      } else
        super.removeChild(name, value);
      
    }

    @Override
    public Base makeProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3575610:
        return getType();
      case -1336663592:
        return getAllowed();
      case -911343192:
        return getAllowed();
      case -147553373:
        return getUsed();
      case 3599293:
        return getUsed();
      default:
        return super.makeProperty(hash, name);
      }

    }

    @Override
    public String[] getTypesForProperty(int hash, String name) throws FHIRException {
      switch (hash) {
      case 3575610:
        /* type */ return new String[] { "CodeableConcept" };
      case -911343192:
        /* allowed */ return new String[] { "unsignedInt", "string", "Money" };
      case 3599293:
        /* used */ return new String[] { "unsignedInt", "Money" };
      default:
        return super.getTypesForProperty(hash, name);
      }

    }

    @Override
    public Base addChild(String name) throws FHIRException {
      if (name.equals("type")) {
        this.type = new CodeableConcept();
        return this.type;
      } else if (name.equals("allowedUnsignedInt")) {
        this.allowed = new UnsignedIntType();
        return this.allowed;
      } else if (name.equals("allowedString")) {
        this.allowed = new StringType();
        return this.allowed;
      } else if (name.equals("allowedMoney")) {
        this.allowed = new Money();
        return this.allowed;
      } else if (name.equals("usedUnsignedInt")) {
        this.used = new UnsignedIntType();
        return this.used;
      } else if (name.equals("usedMoney")) {
        this.used = new Money();
        return this.used;
      } else
        return super.addChild(name);
    }

    public BenefitComponent copy() {
      BenefitComponent dst = new BenefitComponent();
      copyValues(dst);
      return dst;
    }

    public void copyValues(BenefitComponent dst) {
      super.copyValues(dst);
      dst.type = type == null ? null : type.copy();
      dst.allowed = allowed == null ? null : allowed.copy();
      dst.used = used == null ? null : used.copy();
    }

    @Override
    public boolean equalsDeep(Base other_) {
      if (!super.equalsDeep(other_))
        return false;
      if (!(other_ instanceof BenefitComponent))
        return false;
      BenefitComponent o = (BenefitComponent) other_;
      return compareDeep(type, o.type, true) && compareDeep(allowed, o.allowed, true)
          && compareDeep(used, o.used, true);
    }

    @Override
    public boolean equalsShallow(Base other_) {
      if (!super.equalsShallow(other_))
        return false;
      if (!(other_ instanceof BenefitComponent))
        return false;
      BenefitComponent o = (BenefitComponent) other_;
      return true;
    }

    public boolean isEmpty() {
      return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(type, allowed, used);
    }

    public String fhirType() {
      return "ExplanationOfBenefit.benefitBalance.financial";

    }

  }

  /**
   * A unique identifier assigned to this explanation of benefit.
   */
  @Child(name = "identifier", type = {
      Identifier.class }, order = 0, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Business Identifier for the resource", formalDefinition = "A unique identifier assigned to this explanation of benefit.")
  protected List<Identifier> identifier;

  /**
   * The status of the resource instance.
   */
  @Child(name = "status", type = { CodeType.class }, order = 1, min = 1, max = 1, modifier = true, summary = true)
  @Description(shortDefinition = "active | cancelled | draft | entered-in-error", formalDefinition = "The status of the resource instance.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/explanationofbenefit-status")
  protected Enumeration<ExplanationOfBenefitStatus> status;

  /**
   * The category of claim, e.g. oral, pharmacy, vision, institutional,
   * professional.
   */
  @Child(name = "type", type = { CodeableConcept.class }, order = 2, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "Category or discipline", formalDefinition = "The category of claim, e.g. oral, pharmacy, vision, institutional, professional.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-type")
  protected CodeableConcept type;

  /**
   * A finer grained suite of claim type codes which may convey additional
   * information such as Inpatient vs Outpatient and/or a specialty service.
   */
  @Child(name = "subType", type = {
      CodeableConcept.class }, order = 3, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "More granular claim type", formalDefinition = "A finer grained suite of claim type codes which may convey additional information such as Inpatient vs Outpatient and/or a specialty service.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-subtype")
  protected CodeableConcept subType;

  /**
   * A code to indicate whether the nature of the request is: to request
   * adjudication of products and services previously rendered; or requesting
   * authorization and adjudication for provision in the future; or requesting the
   * non-binding adjudication of the listed products and services which could be
   * provided in the future.
   */
  @Child(name = "use", type = { CodeType.class }, order = 4, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "claim | preauthorization | predetermination", formalDefinition = "A code to indicate whether the nature of the request is: to request adjudication of products and services previously rendered; or requesting authorization and adjudication for provision in the future; or requesting the non-binding adjudication of the listed products and services which could be provided in the future.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/claim-use")
  protected Enumeration<Use> use;

  /**
   * The party to whom the professional services and/or products have been
   * supplied or are being considered and for whom actual for forecast
   * reimbursement is sought.
   */
  @Child(name = "patient", type = { Patient.class }, order = 5, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "The recipient of the products and services", formalDefinition = "The party to whom the professional services and/or products have been supplied or are being considered and for whom actual for forecast reimbursement is sought.")
  protected Reference patient;

  /**
   * The period for which charges are being submitted.
   */
  @Child(name = "billablePeriod", type = {
      Period.class }, order = 6, min = 0, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "Relevant time frame for the claim", formalDefinition = "The period for which charges are being submitted.")
  protected Period billablePeriod;

  /**
   * The date this resource was created.
   */
  @Child(name = "created", type = { DateTimeType.class }, order = 7, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "Response creation date", formalDefinition = "The date this resource was created.")
  protected DateTimeType created;

  /**
   * Individual who created the claim, predetermination or preauthorization.
   */
  @Child(name = "enterer", type = { Practitioner.class,
      PractitionerRole.class }, order = 8, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Author of the claim", formalDefinition = "Individual who created the claim, predetermination or preauthorization.")
  protected Reference enterer;

  /**
   * The party responsible for authorization, adjudication and reimbursement.
   */
  @Child(name = "insurer", type = { Organization.class }, order = 9, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "Party responsible for reimbursement", formalDefinition = "The party responsible for authorization, adjudication and reimbursement.")
  protected Reference insurer;

  /**
   * The provider which is responsible for the claim, predetermination or
   * preauthorization.
   */
  @Child(name = "provider", type = { Practitioner.class, PractitionerRole.class,
      Organization.class }, order = 10, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "Party responsible for the claim", formalDefinition = "The provider which is responsible for the claim, predetermination or preauthorization.")
  protected Reference provider;

  /**
   * The provider-required urgency of processing the request. Typical values
   * include: stat, routine deferred.
   */
  @Child(name = "priority", type = {
      CodeableConcept.class }, order = 11, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Desired processing urgency", formalDefinition = "The provider-required urgency of processing the request. Typical values include: stat, routine deferred.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://terminology.hl7.org/CodeSystem/processpriority")
  protected CodeableConcept priority;

  /**
   * A code to indicate whether and for whom funds are to be reserved for future
   * claims.
   */
  @Child(name = "fundsReserveRequested", type = {
      CodeableConcept.class }, order = 12, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "For whom to reserve funds", formalDefinition = "A code to indicate whether and for whom funds are to be reserved for future claims.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/fundsreserve")
  protected CodeableConcept fundsReserveRequested;

  /**
   * A code, used only on a response to a preauthorization, to indicate whether
   * the benefits payable have been reserved and for whom.
   */
  @Child(name = "fundsReserve", type = {
      CodeableConcept.class }, order = 13, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Funds reserved status", formalDefinition = "A code, used only on a response to a preauthorization, to indicate whether the benefits payable have been reserved and for whom.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/fundsreserve")
  protected CodeableConcept fundsReserve;

  /**
   * Other claims which are related to this claim such as prior submissions or
   * claims for related services or for the same event.
   */
  @Child(name = "related", type = {}, order = 14, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Prior or corollary claims", formalDefinition = "Other claims which are related to this claim such as prior submissions or claims for related services or for the same event.")
  protected List<RelatedClaimComponent> related;

  /**
   * Prescription to support the dispensing of pharmacy, device or vision
   * products.
   */
  @Child(name = "prescription", type = { MedicationRequest.class,
      VisionPrescription.class }, order = 15, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Prescription authorizing services or products", formalDefinition = "Prescription to support the dispensing of pharmacy, device or vision products.")
  protected Reference prescription;

  /**
   * Original prescription which has been superseded by this prescription to
   * support the dispensing of pharmacy services, medications or products.
   */
  @Child(name = "originalPrescription", type = {
      MedicationRequest.class }, order = 16, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Original prescription if superceded by fulfiller", formalDefinition = "Original prescription which has been superseded by this prescription to support the dispensing of pharmacy services, medications or products.")
  protected Reference originalPrescription;

  /**
   * The party to be reimbursed for cost of the products and services according to
   * the terms of the policy.
   */
  @Child(name = "payee", type = {}, order = 17, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Recipient of benefits payable", formalDefinition = "The party to be reimbursed for cost of the products and services according to the terms of the policy.")
  protected PayeeComponent payee;

  /**
   * A reference to a referral resource.
   */
  @Child(name = "referral", type = {
      ServiceRequest.class }, order = 18, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Treatment Referral", formalDefinition = "A reference to a referral resource.")
  protected Reference referral;

  /**
   * Facility where the services were provided.
   */
  @Child(name = "facility", type = { Location.class }, order = 19, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Servicing Facility", formalDefinition = "Facility where the services were provided.")
  protected Reference facility;

  /**
   * The business identifier for the instance of the adjudication request: claim
   * predetermination or preauthorization.
   */
  @Child(name = "claim", type = { Claim.class }, order = 20, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Claim reference", formalDefinition = "The business identifier for the instance of the adjudication request: claim predetermination or preauthorization.")
  protected Reference claim;

  /**
   * The business identifier for the instance of the adjudication response: claim,
   * predetermination or preauthorization response.
   */
  @Child(name = "claimResponse", type = {
      ClaimResponse.class }, order = 21, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Claim response reference", formalDefinition = "The business identifier for the instance of the adjudication response: claim, predetermination or preauthorization response.")
  protected Reference claimResponse;

  /**
   * The outcome of the claim, predetermination, or preauthorization processing.
   */
  @Child(name = "outcome", type = { CodeType.class }, order = 22, min = 1, max = 1, modifier = false, summary = true)
  @Description(shortDefinition = "queued | complete | error | partial", formalDefinition = "The outcome of the claim, predetermination, or preauthorization processing.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/remittance-outcome")
  protected Enumeration<RemittanceOutcome> outcome;

  /**
   * A human readable description of the status of the adjudication.
   */
  @Child(name = "disposition", type = {
      StringType.class }, order = 23, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Disposition Message", formalDefinition = "A human readable description of the status of the adjudication.")
  protected StringType disposition;

  /**
   * Reference from the Insurer which is used in later communications which refers
   * to this adjudication.
   */
  @Child(name = "preAuthRef", type = {
      StringType.class }, order = 24, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Preauthorization reference", formalDefinition = "Reference from the Insurer which is used in later communications which refers to this adjudication.")
  protected List<StringType> preAuthRef;

  /**
   * The timeframe during which the supplied preauthorization reference may be
   * quoted on claims to obtain the adjudication as provided.
   */
  @Child(name = "preAuthRefPeriod", type = {
      Period.class }, order = 25, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Preauthorization in-effect period", formalDefinition = "The timeframe during which the supplied preauthorization reference may be quoted on claims to obtain the adjudication as provided.")
  protected List<Period> preAuthRefPeriod;

  /**
   * The members of the team who provided the products and services.
   */
  @Child(name = "careTeam", type = {}, order = 26, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Care Team members", formalDefinition = "The members of the team who provided the products and services.")
  protected List<CareTeamComponent> careTeam;

  /**
   * Additional information codes regarding exceptions, special considerations,
   * the condition, situation, prior or concurrent issues.
   */
  @Child(name = "supportingInfo", type = {}, order = 27, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Supporting information", formalDefinition = "Additional information codes regarding exceptions, special considerations, the condition, situation, prior or concurrent issues.")
  protected List<SupportingInformationComponent> supportingInfo;

  /**
   * Information about diagnoses relevant to the claim items.
   */
  @Child(name = "diagnosis", type = {}, order = 28, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Pertinent diagnosis information", formalDefinition = "Information about diagnoses relevant to the claim items.")
  protected List<DiagnosisComponent> diagnosis;

  /**
   * Procedures performed on the patient relevant to the billing items with the
   * claim.
   */
  @Child(name = "procedure", type = {}, order = 29, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Clinical procedures performed", formalDefinition = "Procedures performed on the patient relevant to the billing items with the claim.")
  protected List<ProcedureComponent> procedure;

  /**
   * This indicates the relative order of a series of EOBs related to different
   * coverages for the same suite of services.
   */
  @Child(name = "precedence", type = {
      PositiveIntType.class }, order = 30, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Precedence (primary, secondary, etc.)", formalDefinition = "This indicates the relative order of a series of EOBs related to different coverages for the same suite of services.")
  protected PositiveIntType precedence;

  /**
   * Financial instruments for reimbursement for the health care products and
   * services specified on the claim.
   */
  @Child(name = "insurance", type = {}, order = 31, min = 1, max = Child.MAX_UNLIMITED, modifier = false, summary = true)
  @Description(shortDefinition = "Patient insurance information", formalDefinition = "Financial instruments for reimbursement for the health care products and services specified on the claim.")
  protected List<InsuranceComponent> insurance;

  /**
   * Details of a accident which resulted in injuries which required the products
   * and services listed in the claim.
   */
  @Child(name = "accident", type = {}, order = 32, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Details of the event", formalDefinition = "Details of a accident which resulted in injuries which required the products and services listed in the claim.")
  protected AccidentComponent accident;

  /**
   * A claim line. Either a simple (a product or service) or a 'group' of details
   * which can also be a simple items or groups of sub-details.
   */
  @Child(name = "item", type = {}, order = 33, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Product or service provided", formalDefinition = "A claim line. Either a simple (a product or service) or a 'group' of details which can also be a simple items or groups of sub-details.")
  protected List<ItemComponent> item;

  /**
   * The first-tier service adjudications for payor added product or service
   * lines.
   */
  @Child(name = "addItem", type = {}, order = 34, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Insurer added line items", formalDefinition = "The first-tier service adjudications for payor added product or service lines.")
  protected List<AddedItemComponent> addItem;

  /**
   * The adjudication results which are presented at the header level rather than
   * at the line-item or add-item levels.
   */
  @Child(name = "adjudication", type = {
      AdjudicationComponent.class }, order = 35, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Header-level adjudication", formalDefinition = "The adjudication results which are presented at the header level rather than at the line-item or add-item levels.")
  protected List<AdjudicationComponent> adjudication;

  /**
   * Categorized monetary totals for the adjudication.
   */
  @Child(name = "total", type = {}, order = 36, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = true)
  @Description(shortDefinition = "Adjudication totals", formalDefinition = "Categorized monetary totals for the adjudication.")
  protected List<TotalComponent> total;

  /**
   * Payment details for the adjudication of the claim.
   */
  @Child(name = "payment", type = {}, order = 37, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Payment Details", formalDefinition = "Payment details for the adjudication of the claim.")
  protected PaymentComponent payment;

  /**
   * A code for the form to be used for printing the content.
   */
  @Child(name = "formCode", type = {
      CodeableConcept.class }, order = 38, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Printed form identifier", formalDefinition = "A code for the form to be used for printing the content.")
  @ca.uhn.fhir.model.api.annotation.Binding(valueSet = "http://hl7.org/fhir/ValueSet/forms")
  protected CodeableConcept formCode;

  /**
   * The actual form, by reference or inclusion, for printing the content or an
   * EOB.
   */
  @Child(name = "form", type = { Attachment.class }, order = 39, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "Printed reference or actual form", formalDefinition = "The actual form, by reference or inclusion, for printing the content or an EOB.")
  protected Attachment form;

  /**
   * A note that describes or explains adjudication results in a human readable
   * form.
   */
  @Child(name = "processNote", type = {}, order = 40, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Note concerning adjudication", formalDefinition = "A note that describes or explains adjudication results in a human readable form.")
  protected List<NoteComponent> processNote;

  /**
   * The term of the benefits documented in this response.
   */
  @Child(name = "benefitPeriod", type = {
      Period.class }, order = 41, min = 0, max = 1, modifier = false, summary = false)
  @Description(shortDefinition = "When the benefits are applicable", formalDefinition = "The term of the benefits documented in this response.")
  protected Period benefitPeriod;

  /**
   * Balance by Benefit Category.
   */
  @Child(name = "benefitBalance", type = {}, order = 42, min = 0, max = Child.MAX_UNLIMITED, modifier = false, summary = false)
  @Description(shortDefinition = "Balance by Benefit Category", formalDefinition = "Balance by Benefit Category.")
  protected List<BenefitBalanceComponent> benefitBalance;

  private static final long serialVersionUID = 1403248694L;

  /**
   * Constructor
   */
  public ExplanationOfBenefit() {
    super();
  }

  /**
   * Constructor
   */
  public ExplanationOfBenefit(ExplanationOfBenefitStatus status, CodeableConcept type, Use use, Reference patient,
      Date created, Reference insurer, Reference provider, RemittanceOutcome outcome, InsuranceComponent insurance) {
    super();
    this.setStatus(status);
    this.setType(type);
    this.setUse(use);
    this.setPatient(patient);
    this.setCreated(created);
    this.setInsurer(insurer);
    this.setProvider(provider);
    this.setOutcome(outcome);
    this.addInsurance(insurance);
  }

  /**
   * @return {@link #identifier} (A unique identifier assigned to this explanation
   *         of benefit.)
   */
  public List<Identifier> getIdentifier() {
    if (this.identifier == null)
      this.identifier = new ArrayList<Identifier>();
    return this.identifier;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setIdentifier(List<Identifier> theIdentifier) {
    this.identifier = theIdentifier;
    return this;
  }

  public boolean hasIdentifier() {
    if (this.identifier == null)
      return false;
    for (Identifier item : this.identifier)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public Identifier addIdentifier() { // 3
    Identifier t = new Identifier();
    if (this.identifier == null)
      this.identifier = new ArrayList<Identifier>();
    this.identifier.add(t);
    return t;
  }

  public ExplanationOfBenefit addIdentifier(Identifier t) { // 3
    if (t == null)
      return this;
    if (this.identifier == null)
      this.identifier = new ArrayList<Identifier>();
    this.identifier.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #identifier}, creating
   *         it if it does not already exist {3}
   */
  public Identifier getIdentifierFirstRep() {
    if (getIdentifier().isEmpty()) {
      addIdentifier();
    }
    return getIdentifier().get(0);
  }

  /**
   * @return {@link #status} (The status of the resource instance.). This is the
   *         underlying object with id, value and extensions. The accessor
   *         "getStatus" gives direct access to the value
   */
  public Enumeration<ExplanationOfBenefitStatus> getStatusElement() {
    if (this.status == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.status");
      else if (Configuration.doAutoCreate())
        this.status = new Enumeration<ExplanationOfBenefitStatus>(new ExplanationOfBenefitStatusEnumFactory()); // bb
    return this.status;
  }

  public boolean hasStatusElement() {
    return this.status != null && !this.status.isEmpty();
  }

  public boolean hasStatus() {
    return this.status != null && !this.status.isEmpty();
  }

  /**
   * @param value {@link #status} (The status of the resource instance.). This is
   *              the underlying object with id, value and extensions. The
   *              accessor "getStatus" gives direct access to the value
   */
  public ExplanationOfBenefit setStatusElement(Enumeration<ExplanationOfBenefitStatus> value) {
    this.status = value;
    return this;
  }

  /**
   * @return The status of the resource instance.
   */
  public ExplanationOfBenefitStatus getStatus() {
    return this.status == null ? null : this.status.getValue();
  }

  /**
   * @param value The status of the resource instance.
   */
  public ExplanationOfBenefit setStatus(ExplanationOfBenefitStatus value) {
    if (this.status == null)
      this.status = new Enumeration<ExplanationOfBenefitStatus>(new ExplanationOfBenefitStatusEnumFactory());
    this.status.setValue(value);
    return this;
  }

  /**
   * @return {@link #type} (The category of claim, e.g. oral, pharmacy, vision,
   *         institutional, professional.)
   */
  public CodeableConcept getType() {
    if (this.type == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.type");
      else if (Configuration.doAutoCreate())
        this.type = new CodeableConcept(); // cc
    return this.type;
  }

  public boolean hasType() {
    return this.type != null && !this.type.isEmpty();
  }

  /**
   * @param value {@link #type} (The category of claim, e.g. oral, pharmacy,
   *              vision, institutional, professional.)
   */
  public ExplanationOfBenefit setType(CodeableConcept value) {
    this.type = value;
    return this;
  }

  /**
   * @return {@link #subType} (A finer grained suite of claim type codes which may
   *         convey additional information such as Inpatient vs Outpatient and/or
   *         a specialty service.)
   */
  public CodeableConcept getSubType() {
    if (this.subType == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.subType");
      else if (Configuration.doAutoCreate())
        this.subType = new CodeableConcept(); // cc
    return this.subType;
  }

  public boolean hasSubType() {
    return this.subType != null && !this.subType.isEmpty();
  }

  /**
   * @param value {@link #subType} (A finer grained suite of claim type codes
   *              which may convey additional information such as Inpatient vs
   *              Outpatient and/or a specialty service.)
   */
  public ExplanationOfBenefit setSubType(CodeableConcept value) {
    this.subType = value;
    return this;
  }

  /**
   * @return {@link #use} (A code to indicate whether the nature of the request
   *         is: to request adjudication of products and services previously
   *         rendered; or requesting authorization and adjudication for provision
   *         in the future; or requesting the non-binding adjudication of the
   *         listed products and services which could be provided in the future.).
   *         This is the underlying object with id, value and extensions. The
   *         accessor "getUse" gives direct access to the value
   */
  public Enumeration<Use> getUseElement() {
    if (this.use == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.use");
      else if (Configuration.doAutoCreate())
        this.use = new Enumeration<Use>(new UseEnumFactory()); // bb
    return this.use;
  }

  public boolean hasUseElement() {
    return this.use != null && !this.use.isEmpty();
  }

  public boolean hasUse() {
    return this.use != null && !this.use.isEmpty();
  }

  /**
   * @param value {@link #use} (A code to indicate whether the nature of the
   *              request is: to request adjudication of products and services
   *              previously rendered; or requesting authorization and
   *              adjudication for provision in the future; or requesting the
   *              non-binding adjudication of the listed products and services
   *              which could be provided in the future.). This is the underlying
   *              object with id, value and extensions. The accessor "getUse"
   *              gives direct access to the value
   */
  public ExplanationOfBenefit setUseElement(Enumeration<Use> value) {
    this.use = value;
    return this;
  }

  /**
   * @return A code to indicate whether the nature of the request is: to request
   *         adjudication of products and services previously rendered; or
   *         requesting authorization and adjudication for provision in the
   *         future; or requesting the non-binding adjudication of the listed
   *         products and services which could be provided in the future.
   */
  public Use getUse() {
    return this.use == null ? null : this.use.getValue();
  }

  /**
   * @param value A code to indicate whether the nature of the request is: to
   *              request adjudication of products and services previously
   *              rendered; or requesting authorization and adjudication for
   *              provision in the future; or requesting the non-binding
   *              adjudication of the listed products and services which could be
   *              provided in the future.
   */
  public ExplanationOfBenefit setUse(Use value) {
    if (this.use == null)
      this.use = new Enumeration<Use>(new UseEnumFactory());
    this.use.setValue(value);
    return this;
  }

  /**
   * @return {@link #patient} (The party to whom the professional services and/or
   *         products have been supplied or are being considered and for whom
   *         actual for forecast reimbursement is sought.)
   */
  public Reference getPatient() {
    if (this.patient == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.patient");
      else if (Configuration.doAutoCreate())
        this.patient = new Reference(); // cc
    return this.patient;
  }

  public boolean hasPatient() {
    return this.patient != null && !this.patient.isEmpty();
  }

  /**
   * @param value {@link #patient} (The party to whom the professional services
   *              and/or products have been supplied or are being considered and
   *              for whom actual for forecast reimbursement is sought.)
   */
  public ExplanationOfBenefit setPatient(Reference value) {
    this.patient = value;
    return this;
  }

  /**
   * @return {@link #billablePeriod} (The period for which charges are being
   *         submitted.)
   */
  public Period getBillablePeriod() {
    if (this.billablePeriod == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.billablePeriod");
      else if (Configuration.doAutoCreate())
        this.billablePeriod = new Period(); // cc
    return this.billablePeriod;
  }

  public boolean hasBillablePeriod() {
    return this.billablePeriod != null && !this.billablePeriod.isEmpty();
  }

  /**
   * @param value {@link #billablePeriod} (The period for which charges are being
   *              submitted.)
   */
  public ExplanationOfBenefit setBillablePeriod(Period value) {
    this.billablePeriod = value;
    return this;
  }

  /**
   * @return {@link #created} (The date this resource was created.). This is the
   *         underlying object with id, value and extensions. The accessor
   *         "getCreated" gives direct access to the value
   */
  public DateTimeType getCreatedElement() {
    if (this.created == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.created");
      else if (Configuration.doAutoCreate())
        this.created = new DateTimeType(); // bb
    return this.created;
  }

  public boolean hasCreatedElement() {
    return this.created != null && !this.created.isEmpty();
  }

  public boolean hasCreated() {
    return this.created != null && !this.created.isEmpty();
  }

  /**
   * @param value {@link #created} (The date this resource was created.). This is
   *              the underlying object with id, value and extensions. The
   *              accessor "getCreated" gives direct access to the value
   */
  public ExplanationOfBenefit setCreatedElement(DateTimeType value) {
    this.created = value;
    return this;
  }

  /**
   * @return The date this resource was created.
   */
  public Date getCreated() {
    return this.created == null ? null : this.created.getValue();
  }

  /**
   * @param value The date this resource was created.
   */
  public ExplanationOfBenefit setCreated(Date value) {
    if (this.created == null)
      this.created = new DateTimeType();
    this.created.setValue(value);
    return this;
  }

  /**
   * @return {@link #enterer} (Individual who created the claim, predetermination
   *         or preauthorization.)
   */
  public Reference getEnterer() {
    if (this.enterer == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.enterer");
      else if (Configuration.doAutoCreate())
        this.enterer = new Reference(); // cc
    return this.enterer;
  }

  public boolean hasEnterer() {
    return this.enterer != null && !this.enterer.isEmpty();
  }

  /**
   * @param value {@link #enterer} (Individual who created the claim,
   *              predetermination or preauthorization.)
   */
  public ExplanationOfBenefit setEnterer(Reference value) {
    this.enterer = value;
    return this;
  }

  /**
   * @return {@link #insurer} (The party responsible for authorization,
   *         adjudication and reimbursement.)
   */
  public Reference getInsurer() {
    if (this.insurer == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.insurer");
      else if (Configuration.doAutoCreate())
        this.insurer = new Reference(); // cc
    return this.insurer;
  }

  public boolean hasInsurer() {
    return this.insurer != null && !this.insurer.isEmpty();
  }

  /**
   * @param value {@link #insurer} (The party responsible for authorization,
   *              adjudication and reimbursement.)
   */
  public ExplanationOfBenefit setInsurer(Reference value) {
    this.insurer = value;
    return this;
  }

  /**
   * @return {@link #provider} (The provider which is responsible for the claim,
   *         predetermination or preauthorization.)
   */
  public Reference getProvider() {
    if (this.provider == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.provider");
      else if (Configuration.doAutoCreate())
        this.provider = new Reference(); // cc
    return this.provider;
  }

  public boolean hasProvider() {
    return this.provider != null && !this.provider.isEmpty();
  }

  /**
   * @param value {@link #provider} (The provider which is responsible for the
   *              claim, predetermination or preauthorization.)
   */
  public ExplanationOfBenefit setProvider(Reference value) {
    this.provider = value;
    return this;
  }

  /**
   * @return {@link #priority} (The provider-required urgency of processing the
   *         request. Typical values include: stat, routine deferred.)
   */
  public CodeableConcept getPriority() {
    if (this.priority == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.priority");
      else if (Configuration.doAutoCreate())
        this.priority = new CodeableConcept(); // cc
    return this.priority;
  }

  public boolean hasPriority() {
    return this.priority != null && !this.priority.isEmpty();
  }

  /**
   * @param value {@link #priority} (The provider-required urgency of processing
   *              the request. Typical values include: stat, routine deferred.)
   */
  public ExplanationOfBenefit setPriority(CodeableConcept value) {
    this.priority = value;
    return this;
  }

  /**
   * @return {@link #fundsReserveRequested} (A code to indicate whether and for
   *         whom funds are to be reserved for future claims.)
   */
  public CodeableConcept getFundsReserveRequested() {
    if (this.fundsReserveRequested == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.fundsReserveRequested");
      else if (Configuration.doAutoCreate())
        this.fundsReserveRequested = new CodeableConcept(); // cc
    return this.fundsReserveRequested;
  }

  public boolean hasFundsReserveRequested() {
    return this.fundsReserveRequested != null && !this.fundsReserveRequested.isEmpty();
  }

  /**
   * @param value {@link #fundsReserveRequested} (A code to indicate whether and
   *              for whom funds are to be reserved for future claims.)
   */
  public ExplanationOfBenefit setFundsReserveRequested(CodeableConcept value) {
    this.fundsReserveRequested = value;
    return this;
  }

  /**
   * @return {@link #fundsReserve} (A code, used only on a response to a
   *         preauthorization, to indicate whether the benefits payable have been
   *         reserved and for whom.)
   */
  public CodeableConcept getFundsReserve() {
    if (this.fundsReserve == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.fundsReserve");
      else if (Configuration.doAutoCreate())
        this.fundsReserve = new CodeableConcept(); // cc
    return this.fundsReserve;
  }

  public boolean hasFundsReserve() {
    return this.fundsReserve != null && !this.fundsReserve.isEmpty();
  }

  /**
   * @param value {@link #fundsReserve} (A code, used only on a response to a
   *              preauthorization, to indicate whether the benefits payable have
   *              been reserved and for whom.)
   */
  public ExplanationOfBenefit setFundsReserve(CodeableConcept value) {
    this.fundsReserve = value;
    return this;
  }

  /**
   * @return {@link #related} (Other claims which are related to this claim such
   *         as prior submissions or claims for related services or for the same
   *         event.)
   */
  public List<RelatedClaimComponent> getRelated() {
    if (this.related == null)
      this.related = new ArrayList<RelatedClaimComponent>();
    return this.related;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setRelated(List<RelatedClaimComponent> theRelated) {
    this.related = theRelated;
    return this;
  }

  public boolean hasRelated() {
    if (this.related == null)
      return false;
    for (RelatedClaimComponent item : this.related)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public RelatedClaimComponent addRelated() { // 3
    RelatedClaimComponent t = new RelatedClaimComponent();
    if (this.related == null)
      this.related = new ArrayList<RelatedClaimComponent>();
    this.related.add(t);
    return t;
  }

  public ExplanationOfBenefit addRelated(RelatedClaimComponent t) { // 3
    if (t == null)
      return this;
    if (this.related == null)
      this.related = new ArrayList<RelatedClaimComponent>();
    this.related.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #related}, creating it
   *         if it does not already exist {3}
   */
  public RelatedClaimComponent getRelatedFirstRep() {
    if (getRelated().isEmpty()) {
      addRelated();
    }
    return getRelated().get(0);
  }

  /**
   * @return {@link #prescription} (Prescription to support the dispensing of
   *         pharmacy, device or vision products.)
   */
  public Reference getPrescription() {
    if (this.prescription == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.prescription");
      else if (Configuration.doAutoCreate())
        this.prescription = new Reference(); // cc
    return this.prescription;
  }

  public boolean hasPrescription() {
    return this.prescription != null && !this.prescription.isEmpty();
  }

  /**
   * @param value {@link #prescription} (Prescription to support the dispensing of
   *              pharmacy, device or vision products.)
   */
  public ExplanationOfBenefit setPrescription(Reference value) {
    this.prescription = value;
    return this;
  }

  /**
   * @return {@link #originalPrescription} (Original prescription which has been
   *         superseded by this prescription to support the dispensing of pharmacy
   *         services, medications or products.)
   */
  public Reference getOriginalPrescription() {
    if (this.originalPrescription == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.originalPrescription");
      else if (Configuration.doAutoCreate())
        this.originalPrescription = new Reference(); // cc
    return this.originalPrescription;
  }

  public boolean hasOriginalPrescription() {
    return this.originalPrescription != null && !this.originalPrescription.isEmpty();
  }

  /**
   * @param value {@link #originalPrescription} (Original prescription which has
   *              been superseded by this prescription to support the dispensing
   *              of pharmacy services, medications or products.)
   */
  public ExplanationOfBenefit setOriginalPrescription(Reference value) {
    this.originalPrescription = value;
    return this;
  }

  /**
   * @return {@link #payee} (The party to be reimbursed for cost of the products
   *         and services according to the terms of the policy.)
   */
  public PayeeComponent getPayee() {
    if (this.payee == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.payee");
      else if (Configuration.doAutoCreate())
        this.payee = new PayeeComponent(); // cc
    return this.payee;
  }

  public boolean hasPayee() {
    return this.payee != null && !this.payee.isEmpty();
  }

  /**
   * @param value {@link #payee} (The party to be reimbursed for cost of the
   *              products and services according to the terms of the policy.)
   */
  public ExplanationOfBenefit setPayee(PayeeComponent value) {
    this.payee = value;
    return this;
  }

  /**
   * @return {@link #referral} (A reference to a referral resource.)
   */
  public Reference getReferral() {
    if (this.referral == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.referral");
      else if (Configuration.doAutoCreate())
        this.referral = new Reference(); // cc
    return this.referral;
  }

  public boolean hasReferral() {
    return this.referral != null && !this.referral.isEmpty();
  }

  /**
   * @param value {@link #referral} (A reference to a referral resource.)
   */
  public ExplanationOfBenefit setReferral(Reference value) {
    this.referral = value;
    return this;
  }

  /**
   * @return {@link #facility} (Facility where the services were provided.)
   */
  public Reference getFacility() {
    if (this.facility == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.facility");
      else if (Configuration.doAutoCreate())
        this.facility = new Reference(); // cc
    return this.facility;
  }

  public boolean hasFacility() {
    return this.facility != null && !this.facility.isEmpty();
  }

  /**
   * @param value {@link #facility} (Facility where the services were provided.)
   */
  public ExplanationOfBenefit setFacility(Reference value) {
    this.facility = value;
    return this;
  }

  /**
   * @return {@link #claim} (The business identifier for the instance of the
   *         adjudication request: claim predetermination or preauthorization.)
   */
  public Reference getClaim() {
    if (this.claim == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.claim");
      else if (Configuration.doAutoCreate())
        this.claim = new Reference(); // cc
    return this.claim;
  }

  public boolean hasClaim() {
    return this.claim != null && !this.claim.isEmpty();
  }

  /**
   * @param value {@link #claim} (The business identifier for the instance of the
   *              adjudication request: claim predetermination or
   *              preauthorization.)
   */
  public ExplanationOfBenefit setClaim(Reference value) {
    this.claim = value;
    return this;
  }

  /**
   * @return {@link #claimResponse} (The business identifier for the instance of
   *         the adjudication response: claim, predetermination or
   *         preauthorization response.)
   */
  public Reference getClaimResponse() {
    if (this.claimResponse == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.claimResponse");
      else if (Configuration.doAutoCreate())
        this.claimResponse = new Reference(); // cc
    return this.claimResponse;
  }

  public boolean hasClaimResponse() {
    return this.claimResponse != null && !this.claimResponse.isEmpty();
  }

  /**
   * @param value {@link #claimResponse} (The business identifier for the instance
   *              of the adjudication response: claim, predetermination or
   *              preauthorization response.)
   */
  public ExplanationOfBenefit setClaimResponse(Reference value) {
    this.claimResponse = value;
    return this;
  }

  /**
   * @return {@link #outcome} (The outcome of the claim, predetermination, or
   *         preauthorization processing.). This is the underlying object with id,
   *         value and extensions. The accessor "getOutcome" gives direct access
   *         to the value
   */
  public Enumeration<RemittanceOutcome> getOutcomeElement() {
    if (this.outcome == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.outcome");
      else if (Configuration.doAutoCreate())
        this.outcome = new Enumeration<RemittanceOutcome>(new RemittanceOutcomeEnumFactory()); // bb
    return this.outcome;
  }

  public boolean hasOutcomeElement() {
    return this.outcome != null && !this.outcome.isEmpty();
  }

  public boolean hasOutcome() {
    return this.outcome != null && !this.outcome.isEmpty();
  }

  /**
   * @param value {@link #outcome} (The outcome of the claim, predetermination, or
   *              preauthorization processing.). This is the underlying object
   *              with id, value and extensions. The accessor "getOutcome" gives
   *              direct access to the value
   */
  public ExplanationOfBenefit setOutcomeElement(Enumeration<RemittanceOutcome> value) {
    this.outcome = value;
    return this;
  }

  /**
   * @return The outcome of the claim, predetermination, or preauthorization
   *         processing.
   */
  public RemittanceOutcome getOutcome() {
    return this.outcome == null ? null : this.outcome.getValue();
  }

  /**
   * @param value The outcome of the claim, predetermination, or preauthorization
   *              processing.
   */
  public ExplanationOfBenefit setOutcome(RemittanceOutcome value) {
    if (this.outcome == null)
      this.outcome = new Enumeration<RemittanceOutcome>(new RemittanceOutcomeEnumFactory());
    this.outcome.setValue(value);
    return this;
  }

  /**
   * @return {@link #disposition} (A human readable description of the status of
   *         the adjudication.). This is the underlying object with id, value and
   *         extensions. The accessor "getDisposition" gives direct access to the
   *         value
   */
  public StringType getDispositionElement() {
    if (this.disposition == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.disposition");
      else if (Configuration.doAutoCreate())
        this.disposition = new StringType(); // bb
    return this.disposition;
  }

  public boolean hasDispositionElement() {
    return this.disposition != null && !this.disposition.isEmpty();
  }

  public boolean hasDisposition() {
    return this.disposition != null && !this.disposition.isEmpty();
  }

  /**
   * @param value {@link #disposition} (A human readable description of the status
   *              of the adjudication.). This is the underlying object with id,
   *              value and extensions. The accessor "getDisposition" gives direct
   *              access to the value
   */
  public ExplanationOfBenefit setDispositionElement(StringType value) {
    this.disposition = value;
    return this;
  }

  /**
   * @return A human readable description of the status of the adjudication.
   */
  public String getDisposition() {
    return this.disposition == null ? null : this.disposition.getValue();
  }

  /**
   * @param value A human readable description of the status of the adjudication.
   */
  public ExplanationOfBenefit setDisposition(String value) {
    if (Utilities.noString(value))
      this.disposition = null;
    else {
      if (this.disposition == null)
        this.disposition = new StringType();
      this.disposition.setValue(value);
    }
    return this;
  }

  /**
   * @return {@link #preAuthRef} (Reference from the Insurer which is used in
   *         later communications which refers to this adjudication.)
   */
  public List<StringType> getPreAuthRef() {
    if (this.preAuthRef == null)
      this.preAuthRef = new ArrayList<StringType>();
    return this.preAuthRef;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setPreAuthRef(List<StringType> thePreAuthRef) {
    this.preAuthRef = thePreAuthRef;
    return this;
  }

  public boolean hasPreAuthRef() {
    if (this.preAuthRef == null)
      return false;
    for (StringType item : this.preAuthRef)
      if (!item.isEmpty())
        return true;
    return false;
  }

  /**
   * @return {@link #preAuthRef} (Reference from the Insurer which is used in
   *         later communications which refers to this adjudication.)
   */
  public StringType addPreAuthRefElement() {// 2
    StringType t = new StringType();
    if (this.preAuthRef == null)
      this.preAuthRef = new ArrayList<StringType>();
    this.preAuthRef.add(t);
    return t;
  }

  /**
   * @param value {@link #preAuthRef} (Reference from the Insurer which is used in
   *              later communications which refers to this adjudication.)
   */
  public ExplanationOfBenefit addPreAuthRef(String value) { // 1
    StringType t = new StringType();
    t.setValue(value);
    if (this.preAuthRef == null)
      this.preAuthRef = new ArrayList<StringType>();
    this.preAuthRef.add(t);
    return this;
  }

  /**
   * @param value {@link #preAuthRef} (Reference from the Insurer which is used in
   *              later communications which refers to this adjudication.)
   */
  public boolean hasPreAuthRef(String value) {
    if (this.preAuthRef == null)
      return false;
    for (StringType v : this.preAuthRef)
      if (v.getValue().equals(value)) // string
        return true;
    return false;
  }

  /**
   * @return {@link #preAuthRefPeriod} (The timeframe during which the supplied
   *         preauthorization reference may be quoted on claims to obtain the
   *         adjudication as provided.)
   */
  public List<Period> getPreAuthRefPeriod() {
    if (this.preAuthRefPeriod == null)
      this.preAuthRefPeriod = new ArrayList<Period>();
    return this.preAuthRefPeriod;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setPreAuthRefPeriod(List<Period> thePreAuthRefPeriod) {
    this.preAuthRefPeriod = thePreAuthRefPeriod;
    return this;
  }

  public boolean hasPreAuthRefPeriod() {
    if (this.preAuthRefPeriod == null)
      return false;
    for (Period item : this.preAuthRefPeriod)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public Period addPreAuthRefPeriod() { // 3
    Period t = new Period();
    if (this.preAuthRefPeriod == null)
      this.preAuthRefPeriod = new ArrayList<Period>();
    this.preAuthRefPeriod.add(t);
    return t;
  }

  public ExplanationOfBenefit addPreAuthRefPeriod(Period t) { // 3
    if (t == null)
      return this;
    if (this.preAuthRefPeriod == null)
      this.preAuthRefPeriod = new ArrayList<Period>();
    this.preAuthRefPeriod.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #preAuthRefPeriod},
   *         creating it if it does not already exist {3}
   */
  public Period getPreAuthRefPeriodFirstRep() {
    if (getPreAuthRefPeriod().isEmpty()) {
      addPreAuthRefPeriod();
    }
    return getPreAuthRefPeriod().get(0);
  }

  /**
   * @return {@link #careTeam} (The members of the team who provided the products
   *         and services.)
   */
  public List<CareTeamComponent> getCareTeam() {
    if (this.careTeam == null)
      this.careTeam = new ArrayList<CareTeamComponent>();
    return this.careTeam;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setCareTeam(List<CareTeamComponent> theCareTeam) {
    this.careTeam = theCareTeam;
    return this;
  }

  public boolean hasCareTeam() {
    if (this.careTeam == null)
      return false;
    for (CareTeamComponent item : this.careTeam)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public CareTeamComponent addCareTeam() { // 3
    CareTeamComponent t = new CareTeamComponent();
    if (this.careTeam == null)
      this.careTeam = new ArrayList<CareTeamComponent>();
    this.careTeam.add(t);
    return t;
  }

  public ExplanationOfBenefit addCareTeam(CareTeamComponent t) { // 3
    if (t == null)
      return this;
    if (this.careTeam == null)
      this.careTeam = new ArrayList<CareTeamComponent>();
    this.careTeam.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #careTeam}, creating
   *         it if it does not already exist {3}
   */
  public CareTeamComponent getCareTeamFirstRep() {
    if (getCareTeam().isEmpty()) {
      addCareTeam();
    }
    return getCareTeam().get(0);
  }

  /**
   * @return {@link #supportingInfo} (Additional information codes regarding
   *         exceptions, special considerations, the condition, situation, prior
   *         or concurrent issues.)
   */
  public List<SupportingInformationComponent> getSupportingInfo() {
    if (this.supportingInfo == null)
      this.supportingInfo = new ArrayList<SupportingInformationComponent>();
    return this.supportingInfo;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setSupportingInfo(List<SupportingInformationComponent> theSupportingInfo) {
    this.supportingInfo = theSupportingInfo;
    return this;
  }

  public boolean hasSupportingInfo() {
    if (this.supportingInfo == null)
      return false;
    for (SupportingInformationComponent item : this.supportingInfo)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public SupportingInformationComponent addSupportingInfo() { // 3
    SupportingInformationComponent t = new SupportingInformationComponent();
    if (this.supportingInfo == null)
      this.supportingInfo = new ArrayList<SupportingInformationComponent>();
    this.supportingInfo.add(t);
    return t;
  }

  public ExplanationOfBenefit addSupportingInfo(SupportingInformationComponent t) { // 3
    if (t == null)
      return this;
    if (this.supportingInfo == null)
      this.supportingInfo = new ArrayList<SupportingInformationComponent>();
    this.supportingInfo.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #supportingInfo},
   *         creating it if it does not already exist {3}
   */
  public SupportingInformationComponent getSupportingInfoFirstRep() {
    if (getSupportingInfo().isEmpty()) {
      addSupportingInfo();
    }
    return getSupportingInfo().get(0);
  }

  /**
   * @return {@link #diagnosis} (Information about diagnoses relevant to the claim
   *         items.)
   */
  public List<DiagnosisComponent> getDiagnosis() {
    if (this.diagnosis == null)
      this.diagnosis = new ArrayList<DiagnosisComponent>();
    return this.diagnosis;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setDiagnosis(List<DiagnosisComponent> theDiagnosis) {
    this.diagnosis = theDiagnosis;
    return this;
  }

  public boolean hasDiagnosis() {
    if (this.diagnosis == null)
      return false;
    for (DiagnosisComponent item : this.diagnosis)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public DiagnosisComponent addDiagnosis() { // 3
    DiagnosisComponent t = new DiagnosisComponent();
    if (this.diagnosis == null)
      this.diagnosis = new ArrayList<DiagnosisComponent>();
    this.diagnosis.add(t);
    return t;
  }

  public ExplanationOfBenefit addDiagnosis(DiagnosisComponent t) { // 3
    if (t == null)
      return this;
    if (this.diagnosis == null)
      this.diagnosis = new ArrayList<DiagnosisComponent>();
    this.diagnosis.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #diagnosis}, creating
   *         it if it does not already exist {3}
   */
  public DiagnosisComponent getDiagnosisFirstRep() {
    if (getDiagnosis().isEmpty()) {
      addDiagnosis();
    }
    return getDiagnosis().get(0);
  }

  /**
   * @return {@link #procedure} (Procedures performed on the patient relevant to
   *         the billing items with the claim.)
   */
  public List<ProcedureComponent> getProcedure() {
    if (this.procedure == null)
      this.procedure = new ArrayList<ProcedureComponent>();
    return this.procedure;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setProcedure(List<ProcedureComponent> theProcedure) {
    this.procedure = theProcedure;
    return this;
  }

  public boolean hasProcedure() {
    if (this.procedure == null)
      return false;
    for (ProcedureComponent item : this.procedure)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public ProcedureComponent addProcedure() { // 3
    ProcedureComponent t = new ProcedureComponent();
    if (this.procedure == null)
      this.procedure = new ArrayList<ProcedureComponent>();
    this.procedure.add(t);
    return t;
  }

  public ExplanationOfBenefit addProcedure(ProcedureComponent t) { // 3
    if (t == null)
      return this;
    if (this.procedure == null)
      this.procedure = new ArrayList<ProcedureComponent>();
    this.procedure.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #procedure}, creating
   *         it if it does not already exist {3}
   */
  public ProcedureComponent getProcedureFirstRep() {
    if (getProcedure().isEmpty()) {
      addProcedure();
    }
    return getProcedure().get(0);
  }

  /**
   * @return {@link #precedence} (This indicates the relative order of a series of
   *         EOBs related to different coverages for the same suite of services.).
   *         This is the underlying object with id, value and extensions. The
   *         accessor "getPrecedence" gives direct access to the value
   */
  public PositiveIntType getPrecedenceElement() {
    if (this.precedence == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.precedence");
      else if (Configuration.doAutoCreate())
        this.precedence = new PositiveIntType(); // bb
    return this.precedence;
  }

  public boolean hasPrecedenceElement() {
    return this.precedence != null && !this.precedence.isEmpty();
  }

  public boolean hasPrecedence() {
    return this.precedence != null && !this.precedence.isEmpty();
  }

  /**
   * @param value {@link #precedence} (This indicates the relative order of a
   *              series of EOBs related to different coverages for the same suite
   *              of services.). This is the underlying object with id, value and
   *              extensions. The accessor "getPrecedence" gives direct access to
   *              the value
   */
  public ExplanationOfBenefit setPrecedenceElement(PositiveIntType value) {
    this.precedence = value;
    return this;
  }

  /**
   * @return This indicates the relative order of a series of EOBs related to
   *         different coverages for the same suite of services.
   */
  public int getPrecedence() {
    return this.precedence == null || this.precedence.isEmpty() ? 0 : this.precedence.getValue();
  }

  /**
   * @param value This indicates the relative order of a series of EOBs related to
   *              different coverages for the same suite of services.
   */
  public ExplanationOfBenefit setPrecedence(int value) {
    if (this.precedence == null)
      this.precedence = new PositiveIntType();
    this.precedence.setValue(value);
    return this;
  }

  /**
   * @return {@link #insurance} (Financial instruments for reimbursement for the
   *         health care products and services specified on the claim.)
   */
  public List<InsuranceComponent> getInsurance() {
    if (this.insurance == null)
      this.insurance = new ArrayList<InsuranceComponent>();
    return this.insurance;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setInsurance(List<InsuranceComponent> theInsurance) {
    this.insurance = theInsurance;
    return this;
  }

  public boolean hasInsurance() {
    if (this.insurance == null)
      return false;
    for (InsuranceComponent item : this.insurance)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public InsuranceComponent addInsurance() { // 3
    InsuranceComponent t = new InsuranceComponent();
    if (this.insurance == null)
      this.insurance = new ArrayList<InsuranceComponent>();
    this.insurance.add(t);
    return t;
  }

  public ExplanationOfBenefit addInsurance(InsuranceComponent t) { // 3
    if (t == null)
      return this;
    if (this.insurance == null)
      this.insurance = new ArrayList<InsuranceComponent>();
    this.insurance.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #insurance}, creating
   *         it if it does not already exist {3}
   */
  public InsuranceComponent getInsuranceFirstRep() {
    if (getInsurance().isEmpty()) {
      addInsurance();
    }
    return getInsurance().get(0);
  }

  /**
   * @return {@link #accident} (Details of a accident which resulted in injuries
   *         which required the products and services listed in the claim.)
   */
  public AccidentComponent getAccident() {
    if (this.accident == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.accident");
      else if (Configuration.doAutoCreate())
        this.accident = new AccidentComponent(); // cc
    return this.accident;
  }

  public boolean hasAccident() {
    return this.accident != null && !this.accident.isEmpty();
  }

  /**
   * @param value {@link #accident} (Details of a accident which resulted in
   *              injuries which required the products and services listed in the
   *              claim.)
   */
  public ExplanationOfBenefit setAccident(AccidentComponent value) {
    this.accident = value;
    return this;
  }

  /**
   * @return {@link #item} (A claim line. Either a simple (a product or service)
   *         or a 'group' of details which can also be a simple items or groups of
   *         sub-details.)
   */
  public List<ItemComponent> getItem() {
    if (this.item == null)
      this.item = new ArrayList<ItemComponent>();
    return this.item;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setItem(List<ItemComponent> theItem) {
    this.item = theItem;
    return this;
  }

  public boolean hasItem() {
    if (this.item == null)
      return false;
    for (ItemComponent item : this.item)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public ItemComponent addItem() { // 3
    ItemComponent t = new ItemComponent();
    if (this.item == null)
      this.item = new ArrayList<ItemComponent>();
    this.item.add(t);
    return t;
  }

  public ExplanationOfBenefit addItem(ItemComponent t) { // 3
    if (t == null)
      return this;
    if (this.item == null)
      this.item = new ArrayList<ItemComponent>();
    this.item.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #item}, creating it if
   *         it does not already exist {3}
   */
  public ItemComponent getItemFirstRep() {
    if (getItem().isEmpty()) {
      addItem();
    }
    return getItem().get(0);
  }

  /**
   * @return {@link #addItem} (The first-tier service adjudications for payor
   *         added product or service lines.)
   */
  public List<AddedItemComponent> getAddItem() {
    if (this.addItem == null)
      this.addItem = new ArrayList<AddedItemComponent>();
    return this.addItem;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setAddItem(List<AddedItemComponent> theAddItem) {
    this.addItem = theAddItem;
    return this;
  }

  public boolean hasAddItem() {
    if (this.addItem == null)
      return false;
    for (AddedItemComponent item : this.addItem)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public AddedItemComponent addAddItem() { // 3
    AddedItemComponent t = new AddedItemComponent();
    if (this.addItem == null)
      this.addItem = new ArrayList<AddedItemComponent>();
    this.addItem.add(t);
    return t;
  }

  public ExplanationOfBenefit addAddItem(AddedItemComponent t) { // 3
    if (t == null)
      return this;
    if (this.addItem == null)
      this.addItem = new ArrayList<AddedItemComponent>();
    this.addItem.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #addItem}, creating it
   *         if it does not already exist {3}
   */
  public AddedItemComponent getAddItemFirstRep() {
    if (getAddItem().isEmpty()) {
      addAddItem();
    }
    return getAddItem().get(0);
  }

  /**
   * @return {@link #adjudication} (The adjudication results which are presented
   *         at the header level rather than at the line-item or add-item levels.)
   */
  public List<AdjudicationComponent> getAdjudication() {
    if (this.adjudication == null)
      this.adjudication = new ArrayList<AdjudicationComponent>();
    return this.adjudication;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setAdjudication(List<AdjudicationComponent> theAdjudication) {
    this.adjudication = theAdjudication;
    return this;
  }

  public boolean hasAdjudication() {
    if (this.adjudication == null)
      return false;
    for (AdjudicationComponent item : this.adjudication)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public AdjudicationComponent addAdjudication() { // 3
    AdjudicationComponent t = new AdjudicationComponent();
    if (this.adjudication == null)
      this.adjudication = new ArrayList<AdjudicationComponent>();
    this.adjudication.add(t);
    return t;
  }

  public ExplanationOfBenefit addAdjudication(AdjudicationComponent t) { // 3
    if (t == null)
      return this;
    if (this.adjudication == null)
      this.adjudication = new ArrayList<AdjudicationComponent>();
    this.adjudication.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #adjudication},
   *         creating it if it does not already exist {3}
   */
  public AdjudicationComponent getAdjudicationFirstRep() {
    if (getAdjudication().isEmpty()) {
      addAdjudication();
    }
    return getAdjudication().get(0);
  }

  /**
   * @return {@link #total} (Categorized monetary totals for the adjudication.)
   */
  public List<TotalComponent> getTotal() {
    if (this.total == null)
      this.total = new ArrayList<TotalComponent>();
    return this.total;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setTotal(List<TotalComponent> theTotal) {
    this.total = theTotal;
    return this;
  }

  public boolean hasTotal() {
    if (this.total == null)
      return false;
    for (TotalComponent item : this.total)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public TotalComponent addTotal() { // 3
    TotalComponent t = new TotalComponent();
    if (this.total == null)
      this.total = new ArrayList<TotalComponent>();
    this.total.add(t);
    return t;
  }

  public ExplanationOfBenefit addTotal(TotalComponent t) { // 3
    if (t == null)
      return this;
    if (this.total == null)
      this.total = new ArrayList<TotalComponent>();
    this.total.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #total}, creating it
   *         if it does not already exist {3}
   */
  public TotalComponent getTotalFirstRep() {
    if (getTotal().isEmpty()) {
      addTotal();
    }
    return getTotal().get(0);
  }

  /**
   * @return {@link #payment} (Payment details for the adjudication of the claim.)
   */
  public PaymentComponent getPayment() {
    if (this.payment == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.payment");
      else if (Configuration.doAutoCreate())
        this.payment = new PaymentComponent(); // cc
    return this.payment;
  }

  public boolean hasPayment() {
    return this.payment != null && !this.payment.isEmpty();
  }

  /**
   * @param value {@link #payment} (Payment details for the adjudication of the
   *              claim.)
   */
  public ExplanationOfBenefit setPayment(PaymentComponent value) {
    this.payment = value;
    return this;
  }

  /**
   * @return {@link #formCode} (A code for the form to be used for printing the
   *         content.)
   */
  public CodeableConcept getFormCode() {
    if (this.formCode == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.formCode");
      else if (Configuration.doAutoCreate())
        this.formCode = new CodeableConcept(); // cc
    return this.formCode;
  }

  public boolean hasFormCode() {
    return this.formCode != null && !this.formCode.isEmpty();
  }

  /**
   * @param value {@link #formCode} (A code for the form to be used for printing
   *              the content.)
   */
  public ExplanationOfBenefit setFormCode(CodeableConcept value) {
    this.formCode = value;
    return this;
  }

  /**
   * @return {@link #form} (The actual form, by reference or inclusion, for
   *         printing the content or an EOB.)
   */
  public Attachment getForm() {
    if (this.form == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.form");
      else if (Configuration.doAutoCreate())
        this.form = new Attachment(); // cc
    return this.form;
  }

  public boolean hasForm() {
    return this.form != null && !this.form.isEmpty();
  }

  /**
   * @param value {@link #form} (The actual form, by reference or inclusion, for
   *              printing the content or an EOB.)
   */
  public ExplanationOfBenefit setForm(Attachment value) {
    this.form = value;
    return this;
  }

  /**
   * @return {@link #processNote} (A note that describes or explains adjudication
   *         results in a human readable form.)
   */
  public List<NoteComponent> getProcessNote() {
    if (this.processNote == null)
      this.processNote = new ArrayList<NoteComponent>();
    return this.processNote;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setProcessNote(List<NoteComponent> theProcessNote) {
    this.processNote = theProcessNote;
    return this;
  }

  public boolean hasProcessNote() {
    if (this.processNote == null)
      return false;
    for (NoteComponent item : this.processNote)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public NoteComponent addProcessNote() { // 3
    NoteComponent t = new NoteComponent();
    if (this.processNote == null)
      this.processNote = new ArrayList<NoteComponent>();
    this.processNote.add(t);
    return t;
  }

  public ExplanationOfBenefit addProcessNote(NoteComponent t) { // 3
    if (t == null)
      return this;
    if (this.processNote == null)
      this.processNote = new ArrayList<NoteComponent>();
    this.processNote.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #processNote},
   *         creating it if it does not already exist {3}
   */
  public NoteComponent getProcessNoteFirstRep() {
    if (getProcessNote().isEmpty()) {
      addProcessNote();
    }
    return getProcessNote().get(0);
  }

  /**
   * @return {@link #benefitPeriod} (The term of the benefits documented in this
   *         response.)
   */
  public Period getBenefitPeriod() {
    if (this.benefitPeriod == null)
      if (Configuration.errorOnAutoCreate())
        throw new Error("Attempt to auto-create ExplanationOfBenefit.benefitPeriod");
      else if (Configuration.doAutoCreate())
        this.benefitPeriod = new Period(); // cc
    return this.benefitPeriod;
  }

  public boolean hasBenefitPeriod() {
    return this.benefitPeriod != null && !this.benefitPeriod.isEmpty();
  }

  /**
   * @param value {@link #benefitPeriod} (The term of the benefits documented in
   *              this response.)
   */
  public ExplanationOfBenefit setBenefitPeriod(Period value) {
    this.benefitPeriod = value;
    return this;
  }

  /**
   * @return {@link #benefitBalance} (Balance by Benefit Category.)
   */
  public List<BenefitBalanceComponent> getBenefitBalance() {
    if (this.benefitBalance == null)
      this.benefitBalance = new ArrayList<BenefitBalanceComponent>();
    return this.benefitBalance;
  }

  /**
   * @return Returns a reference to <code>this</code> for easy method chaining
   */
  public ExplanationOfBenefit setBenefitBalance(List<BenefitBalanceComponent> theBenefitBalance) {
    this.benefitBalance = theBenefitBalance;
    return this;
  }

  public boolean hasBenefitBalance() {
    if (this.benefitBalance == null)
      return false;
    for (BenefitBalanceComponent item : this.benefitBalance)
      if (!item.isEmpty())
        return true;
    return false;
  }

  public BenefitBalanceComponent addBenefitBalance() { // 3
    BenefitBalanceComponent t = new BenefitBalanceComponent();
    if (this.benefitBalance == null)
      this.benefitBalance = new ArrayList<BenefitBalanceComponent>();
    this.benefitBalance.add(t);
    return t;
  }

  public ExplanationOfBenefit addBenefitBalance(BenefitBalanceComponent t) { // 3
    if (t == null)
      return this;
    if (this.benefitBalance == null)
      this.benefitBalance = new ArrayList<BenefitBalanceComponent>();
    this.benefitBalance.add(t);
    return this;
  }

  /**
   * @return The first repetition of repeating field {@link #benefitBalance},
   *         creating it if it does not already exist {3}
   */
  public BenefitBalanceComponent getBenefitBalanceFirstRep() {
    if (getBenefitBalance().isEmpty()) {
      addBenefitBalance();
    }
    return getBenefitBalance().get(0);
  }

  protected void listChildren(List<Property> children) {
    super.listChildren(children);
    children.add(new Property("identifier", "Identifier",
        "A unique identifier assigned to this explanation of benefit.", 0, java.lang.Integer.MAX_VALUE, identifier));
    children.add(new Property("status", "code", "The status of the resource instance.", 0, 1, status));
    children.add(new Property("type", "CodeableConcept",
        "The category of claim, e.g. oral, pharmacy, vision, institutional, professional.", 0, 1, type));
    children.add(new Property("subType", "CodeableConcept",
        "A finer grained suite of claim type codes which may convey additional information such as Inpatient vs Outpatient and/or a specialty service.",
        0, 1, subType));
    children.add(new Property("use", "code",
        "A code to indicate whether the nature of the request is: to request adjudication of products and services previously rendered; or requesting authorization and adjudication for provision in the future; or requesting the non-binding adjudication of the listed products and services which could be provided in the future.",
        0, 1, use));
    children.add(new Property("patient", "Reference(Patient)",
        "The party to whom the professional services and/or products have been supplied or are being considered and for whom actual for forecast reimbursement is sought.",
        0, 1, patient));
    children.add(new Property("billablePeriod", "Period", "The period for which charges are being submitted.", 0, 1,
        billablePeriod));
    children.add(new Property("created", "dateTime", "The date this resource was created.", 0, 1, created));
    children.add(new Property("enterer", "Reference(Practitioner|PractitionerRole)",
        "Individual who created the claim, predetermination or preauthorization.", 0, 1, enterer));
    children.add(new Property("insurer", "Reference(Organization)",
        "The party responsible for authorization, adjudication and reimbursement.", 0, 1, insurer));
    children.add(new Property("provider", "Reference(Practitioner|PractitionerRole|Organization)",
        "The provider which is responsible for the claim, predetermination or preauthorization.", 0, 1, provider));
    children.add(new Property("priority", "CodeableConcept",
        "The provider-required urgency of processing the request. Typical values include: stat, routine deferred.", 0,
        1, priority));
    children.add(new Property("fundsReserveRequested", "CodeableConcept",
        "A code to indicate whether and for whom funds are to be reserved for future claims.", 0, 1,
        fundsReserveRequested));
    children.add(new Property("fundsReserve", "CodeableConcept",
        "A code, used only on a response to a preauthorization, to indicate whether the benefits payable have been reserved and for whom.",
        0, 1, fundsReserve));
    children.add(new Property("related", "",
        "Other claims which are related to this claim such as prior submissions or claims for related services or for the same event.",
        0, java.lang.Integer.MAX_VALUE, related));
    children.add(new Property("prescription", "Reference(MedicationRequest|VisionPrescription)",
        "Prescription to support the dispensing of pharmacy, device or vision products.", 0, 1, prescription));
    children.add(new Property("originalPrescription", "Reference(MedicationRequest)",
        "Original prescription which has been superseded by this prescription to support the dispensing of pharmacy services, medications or products.",
        0, 1, originalPrescription));
    children.add(new Property("payee", "",
        "The party to be reimbursed for cost of the products and services according to the terms of the policy.", 0, 1,
        payee));
    children.add(
        new Property("referral", "Reference(ServiceRequest)", "A reference to a referral resource.", 0, 1, referral));
    children.add(
        new Property("facility", "Reference(Location)", "Facility where the services were provided.", 0, 1, facility));
    children.add(new Property("claim", "Reference(Claim)",
        "The business identifier for the instance of the adjudication request: claim predetermination or preauthorization.",
        0, 1, claim));
    children.add(new Property("claimResponse", "Reference(ClaimResponse)",
        "The business identifier for the instance of the adjudication response: claim, predetermination or preauthorization response.",
        0, 1, claimResponse));
    children.add(new Property("outcome", "code",
        "The outcome of the claim, predetermination, or preauthorization processing.", 0, 1, outcome));
    children.add(new Property("disposition", "string",
        "A human readable description of the status of the adjudication.", 0, 1, disposition));
    children.add(new Property("preAuthRef", "string",
        "Reference from the Insurer which is used in later communications which refers to this adjudication.", 0,
        java.lang.Integer.MAX_VALUE, preAuthRef));
    children.add(new Property("preAuthRefPeriod", "Period",
        "The timeframe during which the supplied preauthorization reference may be quoted on claims to obtain the adjudication as provided.",
        0, java.lang.Integer.MAX_VALUE, preAuthRefPeriod));
    children.add(new Property("careTeam", "", "The members of the team who provided the products and services.", 0,
        java.lang.Integer.MAX_VALUE, careTeam));
    children.add(new Property("supportingInfo", "",
        "Additional information codes regarding exceptions, special considerations, the condition, situation, prior or concurrent issues.",
        0, java.lang.Integer.MAX_VALUE, supportingInfo));
    children.add(new Property("diagnosis", "", "Information about diagnoses relevant to the claim items.", 0,
        java.lang.Integer.MAX_VALUE, diagnosis));
    children.add(new Property("procedure", "",
        "Procedures performed on the patient relevant to the billing items with the claim.", 0,
        java.lang.Integer.MAX_VALUE, procedure));
    children.add(new Property("precedence", "positiveInt",
        "This indicates the relative order of a series of EOBs related to different coverages for the same suite of services.",
        0, 1, precedence));
    children.add(new Property("insurance", "",
        "Financial instruments for reimbursement for the health care products and services specified on the claim.", 0,
        java.lang.Integer.MAX_VALUE, insurance));
    children.add(new Property("accident", "",
        "Details of a accident which resulted in injuries which required the products and services listed in the claim.",
        0, 1, accident));
    children.add(new Property("item", "",
        "A claim line. Either a simple (a product or service) or a 'group' of details which can also be a simple items or groups of sub-details.",
        0, java.lang.Integer.MAX_VALUE, item));
    children.add(
        new Property("addItem", "", "The first-tier service adjudications for payor added product or service lines.", 0,
            java.lang.Integer.MAX_VALUE, addItem));
    children.add(new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
        "The adjudication results which are presented at the header level rather than at the line-item or add-item levels.",
        0, java.lang.Integer.MAX_VALUE, adjudication));
    children.add(new Property("total", "", "Categorized monetary totals for the adjudication.", 0,
        java.lang.Integer.MAX_VALUE, total));
    children.add(new Property("payment", "", "Payment details for the adjudication of the claim.", 0, 1, payment));
    children.add(new Property("formCode", "CodeableConcept", "A code for the form to be used for printing the content.",
        0, 1, formCode));
    children.add(new Property("form", "Attachment",
        "The actual form, by reference or inclusion, for printing the content or an EOB.", 0, 1, form));
    children.add(new Property("processNote", "",
        "A note that describes or explains adjudication results in a human readable form.", 0,
        java.lang.Integer.MAX_VALUE, processNote));
    children.add(new Property("benefitPeriod", "Period", "The term of the benefits documented in this response.", 0, 1,
        benefitPeriod));
    children.add(new Property("benefitBalance", "", "Balance by Benefit Category.", 0, java.lang.Integer.MAX_VALUE,
        benefitBalance));
  }

  @Override
  public Property getNamedProperty(int _hash, String _name, boolean _checkValid) throws FHIRException {
    switch (_hash) {
    case -1618432855:
      /* identifier */ return new Property("identifier", "Identifier",
          "A unique identifier assigned to this explanation of benefit.", 0, java.lang.Integer.MAX_VALUE, identifier);
    case -892481550:
      /* status */ return new Property("status", "code", "The status of the resource instance.", 0, 1, status);
    case 3575610:
      /* type */ return new Property("type", "CodeableConcept",
          "The category of claim, e.g. oral, pharmacy, vision, institutional, professional.", 0, 1, type);
    case -1868521062:
      /* subType */ return new Property("subType", "CodeableConcept",
          "A finer grained suite of claim type codes which may convey additional information such as Inpatient vs Outpatient and/or a specialty service.",
          0, 1, subType);
    case 116103:
      /* use */ return new Property("use", "code",
          "A code to indicate whether the nature of the request is: to request adjudication of products and services previously rendered; or requesting authorization and adjudication for provision in the future; or requesting the non-binding adjudication of the listed products and services which could be provided in the future.",
          0, 1, use);
    case -791418107:
      /* patient */ return new Property("patient", "Reference(Patient)",
          "The party to whom the professional services and/or products have been supplied or are being considered and for whom actual for forecast reimbursement is sought.",
          0, 1, patient);
    case -332066046:
      /* billablePeriod */ return new Property("billablePeriod", "Period",
          "The period for which charges are being submitted.", 0, 1, billablePeriod);
    case 1028554472:
      /* created */ return new Property("created", "dateTime", "The date this resource was created.", 0, 1, created);
    case -1591951995:
      /* enterer */ return new Property("enterer", "Reference(Practitioner|PractitionerRole)",
          "Individual who created the claim, predetermination or preauthorization.", 0, 1, enterer);
    case 1957615864:
      /* insurer */ return new Property("insurer", "Reference(Organization)",
          "The party responsible for authorization, adjudication and reimbursement.", 0, 1, insurer);
    case -987494927:
      /* provider */ return new Property("provider", "Reference(Practitioner|PractitionerRole|Organization)",
          "The provider which is responsible for the claim, predetermination or preauthorization.", 0, 1, provider);
    case -1165461084:
      /* priority */ return new Property("priority", "CodeableConcept",
          "The provider-required urgency of processing the request. Typical values include: stat, routine deferred.", 0,
          1, priority);
    case -1688904576:
      /* fundsReserveRequested */ return new Property("fundsReserveRequested", "CodeableConcept",
          "A code to indicate whether and for whom funds are to be reserved for future claims.", 0, 1,
          fundsReserveRequested);
    case 1314609806:
      /* fundsReserve */ return new Property("fundsReserve", "CodeableConcept",
          "A code, used only on a response to a preauthorization, to indicate whether the benefits payable have been reserved and for whom.",
          0, 1, fundsReserve);
    case 1090493483:
      /* related */ return new Property("related", "",
          "Other claims which are related to this claim such as prior submissions or claims for related services or for the same event.",
          0, java.lang.Integer.MAX_VALUE, related);
    case 460301338:
      /* prescription */ return new Property("prescription", "Reference(MedicationRequest|VisionPrescription)",
          "Prescription to support the dispensing of pharmacy, device or vision products.", 0, 1, prescription);
    case -1814015861:
      /* originalPrescription */ return new Property("originalPrescription", "Reference(MedicationRequest)",
          "Original prescription which has been superseded by this prescription to support the dispensing of pharmacy services, medications or products.",
          0, 1, originalPrescription);
    case 106443592:
      /* payee */ return new Property("payee", "",
          "The party to be reimbursed for cost of the products and services according to the terms of the policy.", 0,
          1, payee);
    case -722568291:
      /* referral */ return new Property("referral", "Reference(ServiceRequest)", "A reference to a referral resource.",
          0, 1, referral);
    case 501116579:
      /* facility */ return new Property("facility", "Reference(Location)",
          "Facility where the services were provided.", 0, 1, facility);
    case 94742588:
      /* claim */ return new Property("claim", "Reference(Claim)",
          "The business identifier for the instance of the adjudication request: claim predetermination or preauthorization.",
          0, 1, claim);
    case 689513629:
      /* claimResponse */ return new Property("claimResponse", "Reference(ClaimResponse)",
          "The business identifier for the instance of the adjudication response: claim, predetermination or preauthorization response.",
          0, 1, claimResponse);
    case -1106507950:
      /* outcome */ return new Property("outcome", "code",
          "The outcome of the claim, predetermination, or preauthorization processing.", 0, 1, outcome);
    case 583380919:
      /* disposition */ return new Property("disposition", "string",
          "A human readable description of the status of the adjudication.", 0, 1, disposition);
    case 522246568:
      /* preAuthRef */ return new Property("preAuthRef", "string",
          "Reference from the Insurer which is used in later communications which refers to this adjudication.", 0,
          java.lang.Integer.MAX_VALUE, preAuthRef);
    case -1262920311:
      /* preAuthRefPeriod */ return new Property("preAuthRefPeriod", "Period",
          "The timeframe during which the supplied preauthorization reference may be quoted on claims to obtain the adjudication as provided.",
          0, java.lang.Integer.MAX_VALUE, preAuthRefPeriod);
    case -7323378:
      /* careTeam */ return new Property("careTeam", "",
          "The members of the team who provided the products and services.", 0, java.lang.Integer.MAX_VALUE, careTeam);
    case 1922406657:
      /* supportingInfo */ return new Property("supportingInfo", "",
          "Additional information codes regarding exceptions, special considerations, the condition, situation, prior or concurrent issues.",
          0, java.lang.Integer.MAX_VALUE, supportingInfo);
    case 1196993265:
      /* diagnosis */ return new Property("diagnosis", "", "Information about diagnoses relevant to the claim items.",
          0, java.lang.Integer.MAX_VALUE, diagnosis);
    case -1095204141:
      /* procedure */ return new Property("procedure", "",
          "Procedures performed on the patient relevant to the billing items with the claim.", 0,
          java.lang.Integer.MAX_VALUE, procedure);
    case 159695370:
      /* precedence */ return new Property("precedence", "positiveInt",
          "This indicates the relative order of a series of EOBs related to different coverages for the same suite of services.",
          0, 1, precedence);
    case 73049818:
      /* insurance */ return new Property("insurance", "",
          "Financial instruments for reimbursement for the health care products and services specified on the claim.",
          0, java.lang.Integer.MAX_VALUE, insurance);
    case -2143202801:
      /* accident */ return new Property("accident", "",
          "Details of a accident which resulted in injuries which required the products and services listed in the claim.",
          0, 1, accident);
    case 3242771:
      /* item */ return new Property("item", "",
          "A claim line. Either a simple (a product or service) or a 'group' of details which can also be a simple items or groups of sub-details.",
          0, java.lang.Integer.MAX_VALUE, item);
    case -1148899500:
      /* addItem */ return new Property("addItem", "",
          "The first-tier service adjudications for payor added product or service lines.", 0,
          java.lang.Integer.MAX_VALUE, addItem);
    case -231349275:
      /* adjudication */ return new Property("adjudication", "@ExplanationOfBenefit.item.adjudication",
          "The adjudication results which are presented at the header level rather than at the line-item or add-item levels.",
          0, java.lang.Integer.MAX_VALUE, adjudication);
    case 110549828:
      /* total */ return new Property("total", "", "Categorized monetary totals for the adjudication.", 0,
          java.lang.Integer.MAX_VALUE, total);
    case -786681338:
      /* payment */ return new Property("payment", "", "Payment details for the adjudication of the claim.", 0, 1,
          payment);
    case 473181393:
      /* formCode */ return new Property("formCode", "CodeableConcept",
          "A code for the form to be used for printing the content.", 0, 1, formCode);
    case 3148996:
      /* form */ return new Property("form", "Attachment",
          "The actual form, by reference or inclusion, for printing the content or an EOB.", 0, 1, form);
    case 202339073:
      /* processNote */ return new Property("processNote", "",
          "A note that describes or explains adjudication results in a human readable form.", 0,
          java.lang.Integer.MAX_VALUE, processNote);
    case -407369416:
      /* benefitPeriod */ return new Property("benefitPeriod", "Period",
          "The term of the benefits documented in this response.", 0, 1, benefitPeriod);
    case 596003397:
      /* benefitBalance */ return new Property("benefitBalance", "", "Balance by Benefit Category.", 0,
          java.lang.Integer.MAX_VALUE, benefitBalance);
    default:
      return super.getNamedProperty(_hash, _name, _checkValid);
    }

  }

  @Override
  public Base[] getProperty(int hash, String name, boolean checkValid) throws FHIRException {
    switch (hash) {
    case -1618432855:
      /* identifier */ return this.identifier == null ? new Base[0]
          : this.identifier.toArray(new Base[this.identifier.size()]); // Identifier
    case -892481550:
      /* status */ return this.status == null ? new Base[0] : new Base[] { this.status }; // Enumeration<ExplanationOfBenefitStatus>
    case 3575610:
      /* type */ return this.type == null ? new Base[0] : new Base[] { this.type }; // CodeableConcept
    case -1868521062:
      /* subType */ return this.subType == null ? new Base[0] : new Base[] { this.subType }; // CodeableConcept
    case 116103:
      /* use */ return this.use == null ? new Base[0] : new Base[] { this.use }; // Enumeration<Use>
    case -791418107:
      /* patient */ return this.patient == null ? new Base[0] : new Base[] { this.patient }; // Reference
    case -332066046:
      /* billablePeriod */ return this.billablePeriod == null ? new Base[0] : new Base[] { this.billablePeriod }; // Period
    case 1028554472:
      /* created */ return this.created == null ? new Base[0] : new Base[] { this.created }; // DateTimeType
    case -1591951995:
      /* enterer */ return this.enterer == null ? new Base[0] : new Base[] { this.enterer }; // Reference
    case 1957615864:
      /* insurer */ return this.insurer == null ? new Base[0] : new Base[] { this.insurer }; // Reference
    case -987494927:
      /* provider */ return this.provider == null ? new Base[0] : new Base[] { this.provider }; // Reference
    case -1165461084:
      /* priority */ return this.priority == null ? new Base[0] : new Base[] { this.priority }; // CodeableConcept
    case -1688904576:
      /* fundsReserveRequested */ return this.fundsReserveRequested == null ? new Base[0]
          : new Base[] { this.fundsReserveRequested }; // CodeableConcept
    case 1314609806:
      /* fundsReserve */ return this.fundsReserve == null ? new Base[0] : new Base[] { this.fundsReserve }; // CodeableConcept
    case 1090493483:
      /* related */ return this.related == null ? new Base[0] : this.related.toArray(new Base[this.related.size()]); // RelatedClaimComponent
    case 460301338:
      /* prescription */ return this.prescription == null ? new Base[0] : new Base[] { this.prescription }; // Reference
    case -1814015861:
      /* originalPrescription */ return this.originalPrescription == null ? new Base[0]
          : new Base[] { this.originalPrescription }; // Reference
    case 106443592:
      /* payee */ return this.payee == null ? new Base[0] : new Base[] { this.payee }; // PayeeComponent
    case -722568291:
      /* referral */ return this.referral == null ? new Base[0] : new Base[] { this.referral }; // Reference
    case 501116579:
      /* facility */ return this.facility == null ? new Base[0] : new Base[] { this.facility }; // Reference
    case 94742588:
      /* claim */ return this.claim == null ? new Base[0] : new Base[] { this.claim }; // Reference
    case 689513629:
      /* claimResponse */ return this.claimResponse == null ? new Base[0] : new Base[] { this.claimResponse }; // Reference
    case -1106507950:
      /* outcome */ return this.outcome == null ? new Base[0] : new Base[] { this.outcome }; // Enumeration<RemittanceOutcome>
    case 583380919:
      /* disposition */ return this.disposition == null ? new Base[0] : new Base[] { this.disposition }; // StringType
    case 522246568:
      /* preAuthRef */ return this.preAuthRef == null ? new Base[0]
          : this.preAuthRef.toArray(new Base[this.preAuthRef.size()]); // StringType
    case -1262920311:
      /* preAuthRefPeriod */ return this.preAuthRefPeriod == null ? new Base[0]
          : this.preAuthRefPeriod.toArray(new Base[this.preAuthRefPeriod.size()]); // Period
    case -7323378:
      /* careTeam */ return this.careTeam == null ? new Base[0] : this.careTeam.toArray(new Base[this.careTeam.size()]); // CareTeamComponent
    case 1922406657:
      /* supportingInfo */ return this.supportingInfo == null ? new Base[0]
          : this.supportingInfo.toArray(new Base[this.supportingInfo.size()]); // SupportingInformationComponent
    case 1196993265:
      /* diagnosis */ return this.diagnosis == null ? new Base[0]
          : this.diagnosis.toArray(new Base[this.diagnosis.size()]); // DiagnosisComponent
    case -1095204141:
      /* procedure */ return this.procedure == null ? new Base[0]
          : this.procedure.toArray(new Base[this.procedure.size()]); // ProcedureComponent
    case 159695370:
      /* precedence */ return this.precedence == null ? new Base[0] : new Base[] { this.precedence }; // PositiveIntType
    case 73049818:
      /* insurance */ return this.insurance == null ? new Base[0]
          : this.insurance.toArray(new Base[this.insurance.size()]); // InsuranceComponent
    case -2143202801:
      /* accident */ return this.accident == null ? new Base[0] : new Base[] { this.accident }; // AccidentComponent
    case 3242771:
      /* item */ return this.item == null ? new Base[0] : this.item.toArray(new Base[this.item.size()]); // ItemComponent
    case -1148899500:
      /* addItem */ return this.addItem == null ? new Base[0] : this.addItem.toArray(new Base[this.addItem.size()]); // AddedItemComponent
    case -231349275:
      /* adjudication */ return this.adjudication == null ? new Base[0]
          : this.adjudication.toArray(new Base[this.adjudication.size()]); // AdjudicationComponent
    case 110549828:
      /* total */ return this.total == null ? new Base[0] : this.total.toArray(new Base[this.total.size()]); // TotalComponent
    case -786681338:
      /* payment */ return this.payment == null ? new Base[0] : new Base[] { this.payment }; // PaymentComponent
    case 473181393:
      /* formCode */ return this.formCode == null ? new Base[0] : new Base[] { this.formCode }; // CodeableConcept
    case 3148996:
      /* form */ return this.form == null ? new Base[0] : new Base[] { this.form }; // Attachment
    case 202339073:
      /* processNote */ return this.processNote == null ? new Base[0]
          : this.processNote.toArray(new Base[this.processNote.size()]); // NoteComponent
    case -407369416:
      /* benefitPeriod */ return this.benefitPeriod == null ? new Base[0] : new Base[] { this.benefitPeriod }; // Period
    case 596003397:
      /* benefitBalance */ return this.benefitBalance == null ? new Base[0]
          : this.benefitBalance.toArray(new Base[this.benefitBalance.size()]); // BenefitBalanceComponent
    default:
      return super.getProperty(hash, name, checkValid);
    }

  }

  @Override
  public Base setProperty(int hash, String name, Base value) throws FHIRException {
    switch (hash) {
    case -1618432855: // identifier
      this.getIdentifier().add(TypeConvertor.castToIdentifier(value)); // Identifier
      return value;
    case -892481550: // status
      value = new ExplanationOfBenefitStatusEnumFactory().fromType(TypeConvertor.castToCode(value));
      this.status = (Enumeration) value; // Enumeration<ExplanationOfBenefitStatus>
      return value;
    case 3575610: // type
      this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      return value;
    case -1868521062: // subType
      this.subType = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      return value;
    case 116103: // use
      value = new UseEnumFactory().fromType(TypeConvertor.castToCode(value));
      this.use = (Enumeration) value; // Enumeration<Use>
      return value;
    case -791418107: // patient
      this.patient = TypeConvertor.castToReference(value); // Reference
      return value;
    case -332066046: // billablePeriod
      this.billablePeriod = TypeConvertor.castToPeriod(value); // Period
      return value;
    case 1028554472: // created
      this.created = TypeConvertor.castToDateTime(value); // DateTimeType
      return value;
    case -1591951995: // enterer
      this.enterer = TypeConvertor.castToReference(value); // Reference
      return value;
    case 1957615864: // insurer
      this.insurer = TypeConvertor.castToReference(value); // Reference
      return value;
    case -987494927: // provider
      this.provider = TypeConvertor.castToReference(value); // Reference
      return value;
    case -1165461084: // priority
      this.priority = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      return value;
    case -1688904576: // fundsReserveRequested
      this.fundsReserveRequested = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      return value;
    case 1314609806: // fundsReserve
      this.fundsReserve = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      return value;
    case 1090493483: // related
      this.getRelated().add((RelatedClaimComponent) value); // RelatedClaimComponent
      return value;
    case 460301338: // prescription
      this.prescription = TypeConvertor.castToReference(value); // Reference
      return value;
    case -1814015861: // originalPrescription
      this.originalPrescription = TypeConvertor.castToReference(value); // Reference
      return value;
    case 106443592: // payee
      this.payee = (PayeeComponent) value; // PayeeComponent
      return value;
    case -722568291: // referral
      this.referral = TypeConvertor.castToReference(value); // Reference
      return value;
    case 501116579: // facility
      this.facility = TypeConvertor.castToReference(value); // Reference
      return value;
    case 94742588: // claim
      this.claim = TypeConvertor.castToReference(value); // Reference
      return value;
    case 689513629: // claimResponse
      this.claimResponse = TypeConvertor.castToReference(value); // Reference
      return value;
    case -1106507950: // outcome
      value = new RemittanceOutcomeEnumFactory().fromType(TypeConvertor.castToCode(value));
      this.outcome = (Enumeration) value; // Enumeration<RemittanceOutcome>
      return value;
    case 583380919: // disposition
      this.disposition = TypeConvertor.castToString(value); // StringType
      return value;
    case 522246568: // preAuthRef
      this.getPreAuthRef().add(TypeConvertor.castToString(value)); // StringType
      return value;
    case -1262920311: // preAuthRefPeriod
      this.getPreAuthRefPeriod().add(TypeConvertor.castToPeriod(value)); // Period
      return value;
    case -7323378: // careTeam
      this.getCareTeam().add((CareTeamComponent) value); // CareTeamComponent
      return value;
    case 1922406657: // supportingInfo
      this.getSupportingInfo().add((SupportingInformationComponent) value); // SupportingInformationComponent
      return value;
    case 1196993265: // diagnosis
      this.getDiagnosis().add((DiagnosisComponent) value); // DiagnosisComponent
      return value;
    case -1095204141: // procedure
      this.getProcedure().add((ProcedureComponent) value); // ProcedureComponent
      return value;
    case 159695370: // precedence
      this.precedence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
      return value;
    case 73049818: // insurance
      this.getInsurance().add((InsuranceComponent) value); // InsuranceComponent
      return value;
    case -2143202801: // accident
      this.accident = (AccidentComponent) value; // AccidentComponent
      return value;
    case 3242771: // item
      this.getItem().add((ItemComponent) value); // ItemComponent
      return value;
    case -1148899500: // addItem
      this.getAddItem().add((AddedItemComponent) value); // AddedItemComponent
      return value;
    case -231349275: // adjudication
      this.getAdjudication().add((AdjudicationComponent) value); // AdjudicationComponent
      return value;
    case 110549828: // total
      this.getTotal().add((TotalComponent) value); // TotalComponent
      return value;
    case -786681338: // payment
      this.payment = (PaymentComponent) value; // PaymentComponent
      return value;
    case 473181393: // formCode
      this.formCode = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
      return value;
    case 3148996: // form
      this.form = TypeConvertor.castToAttachment(value); // Attachment
      return value;
    case 202339073: // processNote
      this.getProcessNote().add((NoteComponent) value); // NoteComponent
      return value;
    case -407369416: // benefitPeriod
      this.benefitPeriod = TypeConvertor.castToPeriod(value); // Period
      return value;
    case 596003397: // benefitBalance
      this.getBenefitBalance().add((BenefitBalanceComponent) value); // BenefitBalanceComponent
      return value;
    default:
      return super.setProperty(hash, name, value);
    }

  }

  @Override
  public Base setProperty(String name, Base value) throws FHIRException {
    if (name.equals("identifier")) {
      this.getIdentifier().add(TypeConvertor.castToIdentifier(value));
    } else if (name.equals("status")) {
      value = new ExplanationOfBenefitStatusEnumFactory().fromType(TypeConvertor.castToCode(value));
      this.status = (Enumeration) value; // Enumeration<ExplanationOfBenefitStatus>
    } else if (name.equals("type")) {
      this.type = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
    } else if (name.equals("subType")) {
      this.subType = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
    } else if (name.equals("use")) {
      value = new UseEnumFactory().fromType(TypeConvertor.castToCode(value));
      this.use = (Enumeration) value; // Enumeration<Use>
    } else if (name.equals("patient")) {
      this.patient = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("billablePeriod")) {
      this.billablePeriod = TypeConvertor.castToPeriod(value); // Period
    } else if (name.equals("created")) {
      this.created = TypeConvertor.castToDateTime(value); // DateTimeType
    } else if (name.equals("enterer")) {
      this.enterer = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("insurer")) {
      this.insurer = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("provider")) {
      this.provider = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("priority")) {
      this.priority = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
    } else if (name.equals("fundsReserveRequested")) {
      this.fundsReserveRequested = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
    } else if (name.equals("fundsReserve")) {
      this.fundsReserve = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
    } else if (name.equals("related")) {
      this.getRelated().add((RelatedClaimComponent) value);
    } else if (name.equals("prescription")) {
      this.prescription = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("originalPrescription")) {
      this.originalPrescription = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("payee")) {
      this.payee = (PayeeComponent) value; // PayeeComponent
    } else if (name.equals("referral")) {
      this.referral = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("facility")) {
      this.facility = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("claim")) {
      this.claim = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("claimResponse")) {
      this.claimResponse = TypeConvertor.castToReference(value); // Reference
    } else if (name.equals("outcome")) {
      value = new RemittanceOutcomeEnumFactory().fromType(TypeConvertor.castToCode(value));
      this.outcome = (Enumeration) value; // Enumeration<RemittanceOutcome>
    } else if (name.equals("disposition")) {
      this.disposition = TypeConvertor.castToString(value); // StringType
    } else if (name.equals("preAuthRef")) {
      this.getPreAuthRef().add(TypeConvertor.castToString(value));
    } else if (name.equals("preAuthRefPeriod")) {
      this.getPreAuthRefPeriod().add(TypeConvertor.castToPeriod(value));
    } else if (name.equals("careTeam")) {
      this.getCareTeam().add((CareTeamComponent) value);
    } else if (name.equals("supportingInfo")) {
      this.getSupportingInfo().add((SupportingInformationComponent) value);
    } else if (name.equals("diagnosis")) {
      this.getDiagnosis().add((DiagnosisComponent) value);
    } else if (name.equals("procedure")) {
      this.getProcedure().add((ProcedureComponent) value);
    } else if (name.equals("precedence")) {
      this.precedence = TypeConvertor.castToPositiveInt(value); // PositiveIntType
    } else if (name.equals("insurance")) {
      this.getInsurance().add((InsuranceComponent) value);
    } else if (name.equals("accident")) {
      this.accident = (AccidentComponent) value; // AccidentComponent
    } else if (name.equals("item")) {
      this.getItem().add((ItemComponent) value);
    } else if (name.equals("addItem")) {
      this.getAddItem().add((AddedItemComponent) value);
    } else if (name.equals("adjudication")) {
      this.getAdjudication().add((AdjudicationComponent) value);
    } else if (name.equals("total")) {
      this.getTotal().add((TotalComponent) value);
    } else if (name.equals("payment")) {
      this.payment = (PaymentComponent) value; // PaymentComponent
    } else if (name.equals("formCode")) {
      this.formCode = TypeConvertor.castToCodeableConcept(value); // CodeableConcept
    } else if (name.equals("form")) {
      this.form = TypeConvertor.castToAttachment(value); // Attachment
    } else if (name.equals("processNote")) {
      this.getProcessNote().add((NoteComponent) value);
    } else if (name.equals("benefitPeriod")) {
      this.benefitPeriod = TypeConvertor.castToPeriod(value); // Period
    } else if (name.equals("benefitBalance")) {
      this.getBenefitBalance().add((BenefitBalanceComponent) value);
    } else
      return super.setProperty(name, value);
    return value;
  }

  @Override
  public void removeChild(String name, Base value) throws FHIRException {
    if (name.equals("identifier")) {
      this.getIdentifier().remove(value);
    } else if (name.equals("status")) {
      this.status = null;
    } else if (name.equals("type")) {
      this.type = null;
    } else if (name.equals("subType")) {
      this.subType = null;
    } else if (name.equals("use")) {
      this.use = null;
    } else if (name.equals("patient")) {
      this.patient = null;
    } else if (name.equals("billablePeriod")) {
      this.billablePeriod = null;
    } else if (name.equals("created")) {
      this.created = null;
    } else if (name.equals("enterer")) {
      this.enterer = null;
    } else if (name.equals("insurer")) {
      this.insurer = null;
    } else if (name.equals("provider")) {
      this.provider = null;
    } else if (name.equals("priority")) {
      this.priority = null;
    } else if (name.equals("fundsReserveRequested")) {
      this.fundsReserveRequested = null;
    } else if (name.equals("fundsReserve")) {
      this.fundsReserve = null;
    } else if (name.equals("related")) {
      this.getRelated().remove((RelatedClaimComponent) value);
    } else if (name.equals("prescription")) {
      this.prescription = null;
    } else if (name.equals("originalPrescription")) {
      this.originalPrescription = null;
    } else if (name.equals("payee")) {
      this.payee = (PayeeComponent) value; // PayeeComponent
    } else if (name.equals("referral")) {
      this.referral = null;
    } else if (name.equals("facility")) {
      this.facility = null;
    } else if (name.equals("claim")) {
      this.claim = null;
    } else if (name.equals("claimResponse")) {
      this.claimResponse = null;
    } else if (name.equals("outcome")) {
      this.outcome = null;
    } else if (name.equals("disposition")) {
      this.disposition = null;
    } else if (name.equals("preAuthRef")) {
      this.getPreAuthRef().remove(value);
    } else if (name.equals("preAuthRefPeriod")) {
      this.getPreAuthRefPeriod().remove(value);
    } else if (name.equals("careTeam")) {
      this.getCareTeam().remove((CareTeamComponent) value);
    } else if (name.equals("supportingInfo")) {
      this.getSupportingInfo().remove((SupportingInformationComponent) value);
    } else if (name.equals("diagnosis")) {
      this.getDiagnosis().remove((DiagnosisComponent) value);
    } else if (name.equals("procedure")) {
      this.getProcedure().remove((ProcedureComponent) value);
    } else if (name.equals("precedence")) {
      this.precedence = null;
    } else if (name.equals("insurance")) {
      this.getInsurance().remove((InsuranceComponent) value);
    } else if (name.equals("accident")) {
      this.accident = (AccidentComponent) value; // AccidentComponent
    } else if (name.equals("item")) {
      this.getItem().remove((ItemComponent) value);
    } else if (name.equals("addItem")) {
      this.getAddItem().remove((AddedItemComponent) value);
    } else if (name.equals("adjudication")) {
      this.getAdjudication().remove((AdjudicationComponent) value);
    } else if (name.equals("total")) {
      this.getTotal().remove((TotalComponent) value);
    } else if (name.equals("payment")) {
      this.payment = (PaymentComponent) value; // PaymentComponent
    } else if (name.equals("formCode")) {
      this.formCode = null;
    } else if (name.equals("form")) {
      this.form = null;
    } else if (name.equals("processNote")) {
      this.getProcessNote().remove((NoteComponent) value);
    } else if (name.equals("benefitPeriod")) {
      this.benefitPeriod = null;
    } else if (name.equals("benefitBalance")) {
      this.getBenefitBalance().remove((BenefitBalanceComponent) value);
    } else
      super.removeChild(name, value);
    
  }

  @Override
  public Base makeProperty(int hash, String name) throws FHIRException {
    switch (hash) {
    case -1618432855:
      return addIdentifier();
    case -892481550:
      return getStatusElement();
    case 3575610:
      return getType();
    case -1868521062:
      return getSubType();
    case 116103:
      return getUseElement();
    case -791418107:
      return getPatient();
    case -332066046:
      return getBillablePeriod();
    case 1028554472:
      return getCreatedElement();
    case -1591951995:
      return getEnterer();
    case 1957615864:
      return getInsurer();
    case -987494927:
      return getProvider();
    case -1165461084:
      return getPriority();
    case -1688904576:
      return getFundsReserveRequested();
    case 1314609806:
      return getFundsReserve();
    case 1090493483:
      return addRelated();
    case 460301338:
      return getPrescription();
    case -1814015861:
      return getOriginalPrescription();
    case 106443592:
      return getPayee();
    case -722568291:
      return getReferral();
    case 501116579:
      return getFacility();
    case 94742588:
      return getClaim();
    case 689513629:
      return getClaimResponse();
    case -1106507950:
      return getOutcomeElement();
    case 583380919:
      return getDispositionElement();
    case 522246568:
      return addPreAuthRefElement();
    case -1262920311:
      return addPreAuthRefPeriod();
    case -7323378:
      return addCareTeam();
    case 1922406657:
      return addSupportingInfo();
    case 1196993265:
      return addDiagnosis();
    case -1095204141:
      return addProcedure();
    case 159695370:
      return getPrecedenceElement();
    case 73049818:
      return addInsurance();
    case -2143202801:
      return getAccident();
    case 3242771:
      return addItem();
    case -1148899500:
      return addAddItem();
    case -231349275:
      return addAdjudication();
    case 110549828:
      return addTotal();
    case -786681338:
      return getPayment();
    case 473181393:
      return getFormCode();
    case 3148996:
      return getForm();
    case 202339073:
      return addProcessNote();
    case -407369416:
      return getBenefitPeriod();
    case 596003397:
      return addBenefitBalance();
    default:
      return super.makeProperty(hash, name);
    }

  }

  @Override
  public String[] getTypesForProperty(int hash, String name) throws FHIRException {
    switch (hash) {
    case -1618432855:
      /* identifier */ return new String[] { "Identifier" };
    case -892481550:
      /* status */ return new String[] { "code" };
    case 3575610:
      /* type */ return new String[] { "CodeableConcept" };
    case -1868521062:
      /* subType */ return new String[] { "CodeableConcept" };
    case 116103:
      /* use */ return new String[] { "code" };
    case -791418107:
      /* patient */ return new String[] { "Reference" };
    case -332066046:
      /* billablePeriod */ return new String[] { "Period" };
    case 1028554472:
      /* created */ return new String[] { "dateTime" };
    case -1591951995:
      /* enterer */ return new String[] { "Reference" };
    case 1957615864:
      /* insurer */ return new String[] { "Reference" };
    case -987494927:
      /* provider */ return new String[] { "Reference" };
    case -1165461084:
      /* priority */ return new String[] { "CodeableConcept" };
    case -1688904576:
      /* fundsReserveRequested */ return new String[] { "CodeableConcept" };
    case 1314609806:
      /* fundsReserve */ return new String[] { "CodeableConcept" };
    case 1090493483:
      /* related */ return new String[] {};
    case 460301338:
      /* prescription */ return new String[] { "Reference" };
    case -1814015861:
      /* originalPrescription */ return new String[] { "Reference" };
    case 106443592:
      /* payee */ return new String[] {};
    case -722568291:
      /* referral */ return new String[] { "Reference" };
    case 501116579:
      /* facility */ return new String[] { "Reference" };
    case 94742588:
      /* claim */ return new String[] { "Reference" };
    case 689513629:
      /* claimResponse */ return new String[] { "Reference" };
    case -1106507950:
      /* outcome */ return new String[] { "code" };
    case 583380919:
      /* disposition */ return new String[] { "string" };
    case 522246568:
      /* preAuthRef */ return new String[] { "string" };
    case -1262920311:
      /* preAuthRefPeriod */ return new String[] { "Period" };
    case -7323378:
      /* careTeam */ return new String[] {};
    case 1922406657:
      /* supportingInfo */ return new String[] {};
    case 1196993265:
      /* diagnosis */ return new String[] {};
    case -1095204141:
      /* procedure */ return new String[] {};
    case 159695370:
      /* precedence */ return new String[] { "positiveInt" };
    case 73049818:
      /* insurance */ return new String[] {};
    case -2143202801:
      /* accident */ return new String[] {};
    case 3242771:
      /* item */ return new String[] {};
    case -1148899500:
      /* addItem */ return new String[] {};
    case -231349275:
      /* adjudication */ return new String[] { "@ExplanationOfBenefit.item.adjudication" };
    case 110549828:
      /* total */ return new String[] {};
    case -786681338:
      /* payment */ return new String[] {};
    case 473181393:
      /* formCode */ return new String[] { "CodeableConcept" };
    case 3148996:
      /* form */ return new String[] { "Attachment" };
    case 202339073:
      /* processNote */ return new String[] {};
    case -407369416:
      /* benefitPeriod */ return new String[] { "Period" };
    case 596003397:
      /* benefitBalance */ return new String[] {};
    default:
      return super.getTypesForProperty(hash, name);
    }

  }

  @Override
  public Base addChild(String name) throws FHIRException {
    if (name.equals("identifier")) {
      return addIdentifier();
    } else if (name.equals("status")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.status");
    } else if (name.equals("type")) {
      this.type = new CodeableConcept();
      return this.type;
    } else if (name.equals("subType")) {
      this.subType = new CodeableConcept();
      return this.subType;
    } else if (name.equals("use")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.use");
    } else if (name.equals("patient")) {
      this.patient = new Reference();
      return this.patient;
    } else if (name.equals("billablePeriod")) {
      this.billablePeriod = new Period();
      return this.billablePeriod;
    } else if (name.equals("created")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.created");
    } else if (name.equals("enterer")) {
      this.enterer = new Reference();
      return this.enterer;
    } else if (name.equals("insurer")) {
      this.insurer = new Reference();
      return this.insurer;
    } else if (name.equals("provider")) {
      this.provider = new Reference();
      return this.provider;
    } else if (name.equals("priority")) {
      this.priority = new CodeableConcept();
      return this.priority;
    } else if (name.equals("fundsReserveRequested")) {
      this.fundsReserveRequested = new CodeableConcept();
      return this.fundsReserveRequested;
    } else if (name.equals("fundsReserve")) {
      this.fundsReserve = new CodeableConcept();
      return this.fundsReserve;
    } else if (name.equals("related")) {
      return addRelated();
    } else if (name.equals("prescription")) {
      this.prescription = new Reference();
      return this.prescription;
    } else if (name.equals("originalPrescription")) {
      this.originalPrescription = new Reference();
      return this.originalPrescription;
    } else if (name.equals("payee")) {
      this.payee = new PayeeComponent();
      return this.payee;
    } else if (name.equals("referral")) {
      this.referral = new Reference();
      return this.referral;
    } else if (name.equals("facility")) {
      this.facility = new Reference();
      return this.facility;
    } else if (name.equals("claim")) {
      this.claim = new Reference();
      return this.claim;
    } else if (name.equals("claimResponse")) {
      this.claimResponse = new Reference();
      return this.claimResponse;
    } else if (name.equals("outcome")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.outcome");
    } else if (name.equals("disposition")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.disposition");
    } else if (name.equals("preAuthRef")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.preAuthRef");
    } else if (name.equals("preAuthRefPeriod")) {
      return addPreAuthRefPeriod();
    } else if (name.equals("careTeam")) {
      return addCareTeam();
    } else if (name.equals("supportingInfo")) {
      return addSupportingInfo();
    } else if (name.equals("diagnosis")) {
      return addDiagnosis();
    } else if (name.equals("procedure")) {
      return addProcedure();
    } else if (name.equals("precedence")) {
      throw new FHIRException("Cannot call addChild on a singleton property ExplanationOfBenefit.precedence");
    } else if (name.equals("insurance")) {
      return addInsurance();
    } else if (name.equals("accident")) {
      this.accident = new AccidentComponent();
      return this.accident;
    } else if (name.equals("item")) {
      return addItem();
    } else if (name.equals("addItem")) {
      return addAddItem();
    } else if (name.equals("adjudication")) {
      return addAdjudication();
    } else if (name.equals("total")) {
      return addTotal();
    } else if (name.equals("payment")) {
      this.payment = new PaymentComponent();
      return this.payment;
    } else if (name.equals("formCode")) {
      this.formCode = new CodeableConcept();
      return this.formCode;
    } else if (name.equals("form")) {
      this.form = new Attachment();
      return this.form;
    } else if (name.equals("processNote")) {
      return addProcessNote();
    } else if (name.equals("benefitPeriod")) {
      this.benefitPeriod = new Period();
      return this.benefitPeriod;
    } else if (name.equals("benefitBalance")) {
      return addBenefitBalance();
    } else
      return super.addChild(name);
  }

  public String fhirType() {
    return "ExplanationOfBenefit";

  }

  public ExplanationOfBenefit copy() {
    ExplanationOfBenefit dst = new ExplanationOfBenefit();
    copyValues(dst);
    return dst;
  }

  public void copyValues(ExplanationOfBenefit dst) {
    super.copyValues(dst);
    if (identifier != null) {
      dst.identifier = new ArrayList<Identifier>();
      for (Identifier i : identifier)
        dst.identifier.add(i.copy());
    }
    ;
    dst.status = status == null ? null : status.copy();
    dst.type = type == null ? null : type.copy();
    dst.subType = subType == null ? null : subType.copy();
    dst.use = use == null ? null : use.copy();
    dst.patient = patient == null ? null : patient.copy();
    dst.billablePeriod = billablePeriod == null ? null : billablePeriod.copy();
    dst.created = created == null ? null : created.copy();
    dst.enterer = enterer == null ? null : enterer.copy();
    dst.insurer = insurer == null ? null : insurer.copy();
    dst.provider = provider == null ? null : provider.copy();
    dst.priority = priority == null ? null : priority.copy();
    dst.fundsReserveRequested = fundsReserveRequested == null ? null : fundsReserveRequested.copy();
    dst.fundsReserve = fundsReserve == null ? null : fundsReserve.copy();
    if (related != null) {
      dst.related = new ArrayList<RelatedClaimComponent>();
      for (RelatedClaimComponent i : related)
        dst.related.add(i.copy());
    }
    ;
    dst.prescription = prescription == null ? null : prescription.copy();
    dst.originalPrescription = originalPrescription == null ? null : originalPrescription.copy();
    dst.payee = payee == null ? null : payee.copy();
    dst.referral = referral == null ? null : referral.copy();
    dst.facility = facility == null ? null : facility.copy();
    dst.claim = claim == null ? null : claim.copy();
    dst.claimResponse = claimResponse == null ? null : claimResponse.copy();
    dst.outcome = outcome == null ? null : outcome.copy();
    dst.disposition = disposition == null ? null : disposition.copy();
    if (preAuthRef != null) {
      dst.preAuthRef = new ArrayList<StringType>();
      for (StringType i : preAuthRef)
        dst.preAuthRef.add(i.copy());
    }
    ;
    if (preAuthRefPeriod != null) {
      dst.preAuthRefPeriod = new ArrayList<Period>();
      for (Period i : preAuthRefPeriod)
        dst.preAuthRefPeriod.add(i.copy());
    }
    ;
    if (careTeam != null) {
      dst.careTeam = new ArrayList<CareTeamComponent>();
      for (CareTeamComponent i : careTeam)
        dst.careTeam.add(i.copy());
    }
    ;
    if (supportingInfo != null) {
      dst.supportingInfo = new ArrayList<SupportingInformationComponent>();
      for (SupportingInformationComponent i : supportingInfo)
        dst.supportingInfo.add(i.copy());
    }
    ;
    if (diagnosis != null) {
      dst.diagnosis = new ArrayList<DiagnosisComponent>();
      for (DiagnosisComponent i : diagnosis)
        dst.diagnosis.add(i.copy());
    }
    ;
    if (procedure != null) {
      dst.procedure = new ArrayList<ProcedureComponent>();
      for (ProcedureComponent i : procedure)
        dst.procedure.add(i.copy());
    }
    ;
    dst.precedence = precedence == null ? null : precedence.copy();
    if (insurance != null) {
      dst.insurance = new ArrayList<InsuranceComponent>();
      for (InsuranceComponent i : insurance)
        dst.insurance.add(i.copy());
    }
    ;
    dst.accident = accident == null ? null : accident.copy();
    if (item != null) {
      dst.item = new ArrayList<ItemComponent>();
      for (ItemComponent i : item)
        dst.item.add(i.copy());
    }
    ;
    if (addItem != null) {
      dst.addItem = new ArrayList<AddedItemComponent>();
      for (AddedItemComponent i : addItem)
        dst.addItem.add(i.copy());
    }
    ;
    if (adjudication != null) {
      dst.adjudication = new ArrayList<AdjudicationComponent>();
      for (AdjudicationComponent i : adjudication)
        dst.adjudication.add(i.copy());
    }
    ;
    if (total != null) {
      dst.total = new ArrayList<TotalComponent>();
      for (TotalComponent i : total)
        dst.total.add(i.copy());
    }
    ;
    dst.payment = payment == null ? null : payment.copy();
    dst.formCode = formCode == null ? null : formCode.copy();
    dst.form = form == null ? null : form.copy();
    if (processNote != null) {
      dst.processNote = new ArrayList<NoteComponent>();
      for (NoteComponent i : processNote)
        dst.processNote.add(i.copy());
    }
    ;
    dst.benefitPeriod = benefitPeriod == null ? null : benefitPeriod.copy();
    if (benefitBalance != null) {
      dst.benefitBalance = new ArrayList<BenefitBalanceComponent>();
      for (BenefitBalanceComponent i : benefitBalance)
        dst.benefitBalance.add(i.copy());
    }
    ;
  }

  protected ExplanationOfBenefit typedCopy() {
    return copy();
  }

  @Override
  public boolean equalsDeep(Base other_) {
    if (!super.equalsDeep(other_))
      return false;
    if (!(other_ instanceof ExplanationOfBenefit))
      return false;
    ExplanationOfBenefit o = (ExplanationOfBenefit) other_;
    return compareDeep(identifier, o.identifier, true) && compareDeep(status, o.status, true)
        && compareDeep(type, o.type, true) && compareDeep(subType, o.subType, true) && compareDeep(use, o.use, true)
        && compareDeep(patient, o.patient, true) && compareDeep(billablePeriod, o.billablePeriod, true)
        && compareDeep(created, o.created, true) && compareDeep(enterer, o.enterer, true)
        && compareDeep(insurer, o.insurer, true) && compareDeep(provider, o.provider, true)
        && compareDeep(priority, o.priority, true) && compareDeep(fundsReserveRequested, o.fundsReserveRequested, true)
        && compareDeep(fundsReserve, o.fundsReserve, true) && compareDeep(related, o.related, true)
        && compareDeep(prescription, o.prescription, true)
        && compareDeep(originalPrescription, o.originalPrescription, true) && compareDeep(payee, o.payee, true)
        && compareDeep(referral, o.referral, true) && compareDeep(facility, o.facility, true)
        && compareDeep(claim, o.claim, true) && compareDeep(claimResponse, o.claimResponse, true)
        && compareDeep(outcome, o.outcome, true) && compareDeep(disposition, o.disposition, true)
        && compareDeep(preAuthRef, o.preAuthRef, true) && compareDeep(preAuthRefPeriod, o.preAuthRefPeriod, true)
        && compareDeep(careTeam, o.careTeam, true) && compareDeep(supportingInfo, o.supportingInfo, true)
        && compareDeep(diagnosis, o.diagnosis, true) && compareDeep(procedure, o.procedure, true)
        && compareDeep(precedence, o.precedence, true) && compareDeep(insurance, o.insurance, true)
        && compareDeep(accident, o.accident, true) && compareDeep(item, o.item, true)
        && compareDeep(addItem, o.addItem, true) && compareDeep(adjudication, o.adjudication, true)
        && compareDeep(total, o.total, true) && compareDeep(payment, o.payment, true)
        && compareDeep(formCode, o.formCode, true) && compareDeep(form, o.form, true)
        && compareDeep(processNote, o.processNote, true) && compareDeep(benefitPeriod, o.benefitPeriod, true)
        && compareDeep(benefitBalance, o.benefitBalance, true);
  }

  @Override
  public boolean equalsShallow(Base other_) {
    if (!super.equalsShallow(other_))
      return false;
    if (!(other_ instanceof ExplanationOfBenefit))
      return false;
    ExplanationOfBenefit o = (ExplanationOfBenefit) other_;
    return compareValues(status, o.status, true) && compareValues(use, o.use, true)
        && compareValues(created, o.created, true) && compareValues(outcome, o.outcome, true)
        && compareValues(disposition, o.disposition, true) && compareValues(preAuthRef, o.preAuthRef, true)
        && compareValues(precedence, o.precedence, true);
  }

  public boolean isEmpty() {
    return super.isEmpty() && ca.uhn.fhir.util.ElementUtil.isEmpty(identifier, status, type, subType, use, patient,
        billablePeriod, created, enterer, insurer, provider, priority, fundsReserveRequested, fundsReserve, related,
        prescription, originalPrescription, payee, referral, facility, claim, claimResponse, outcome, disposition,
        preAuthRef, preAuthRefPeriod, careTeam, supportingInfo, diagnosis, procedure, precedence, insurance, accident,
        item, addItem, adjudication, total, payment, formCode, form, processNote, benefitPeriod, benefitBalance);
  }

  @Override
  public ResourceType getResourceType() {
    return ResourceType.ExplanationOfBenefit;
  }

  /**
   * Search parameter: <b>care-team</b>
   * <p>
   * Description: <b>Member of the CareTeam</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.careTeam.provider</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "care-team", path = "ExplanationOfBenefit.careTeam.provider", description = "Member of the CareTeam", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Practitioner") }, target = {
          Organization.class, Practitioner.class, PractitionerRole.class })
  public static final String SP_CARE_TEAM = "care-team";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>care-team</b>
   * <p>
   * Description: <b>Member of the CareTeam</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.careTeam.provider</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam CARE_TEAM = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_CARE_TEAM);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:care-team</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_CARE_TEAM = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:care-team").toLocked();

  /**
   * Search parameter: <b>claim</b>
   * <p>
   * Description: <b>The reference to the claim</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.claim</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "claim", path = "ExplanationOfBenefit.claim", description = "The reference to the claim", type = "reference", target = {
      Claim.class })
  public static final String SP_CLAIM = "claim";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>claim</b>
   * <p>
   * Description: <b>The reference to the claim</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.claim</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam CLAIM = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_CLAIM);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:claim</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_CLAIM = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:claim").toLocked();

  /**
   * Search parameter: <b>coverage</b>
   * <p>
   * Description: <b>The plan under which the claim was adjudicated</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.insurance.coverage</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "coverage", path = "ExplanationOfBenefit.insurance.coverage", description = "The plan under which the claim was adjudicated", type = "reference", target = {
      Coverage.class })
  public static final String SP_COVERAGE = "coverage";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>coverage</b>
   * <p>
   * Description: <b>The plan under which the claim was adjudicated</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.insurance.coverage</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam COVERAGE = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_COVERAGE);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:coverage</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_COVERAGE = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:coverage").toLocked();

  /**
   * Search parameter: <b>created</b>
   * <p>
   * Description: <b>The creation date for the EOB</b><br>
   * Type: <b>date</b><br>
   * Path: <b>ExplanationOfBenefit.created</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "created", path = "ExplanationOfBenefit.created", description = "The creation date for the EOB", type = "date")
  public static final String SP_CREATED = "created";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>created</b>
   * <p>
   * Description: <b>The creation date for the EOB</b><br>
   * Type: <b>date</b><br>
   * Path: <b>ExplanationOfBenefit.created</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.DateClientParam CREATED = new ca.uhn.fhir.rest.gclient.DateClientParam(
      SP_CREATED);

  /**
   * Search parameter: <b>detail-udi</b>
   * <p>
   * Description: <b>UDI associated with a line item detail product or
   * service</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.detail.udi</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "detail-udi", path = "ExplanationOfBenefit.item.detail.udi", description = "UDI associated with a line item detail product or service", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Device") }, target = {
          Device.class })
  public static final String SP_DETAIL_UDI = "detail-udi";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>detail-udi</b>
   * <p>
   * Description: <b>UDI associated with a line item detail product or
   * service</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.detail.udi</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam DETAIL_UDI = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_DETAIL_UDI);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:detail-udi</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_DETAIL_UDI = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:detail-udi").toLocked();

  /**
   * Search parameter: <b>disposition</b>
   * <p>
   * Description: <b>The contents of the disposition message</b><br>
   * Type: <b>string</b><br>
   * Path: <b>ExplanationOfBenefit.disposition</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "disposition", path = "ExplanationOfBenefit.disposition", description = "The contents of the disposition message", type = "string")
  public static final String SP_DISPOSITION = "disposition";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>disposition</b>
   * <p>
   * Description: <b>The contents of the disposition message</b><br>
   * Type: <b>string</b><br>
   * Path: <b>ExplanationOfBenefit.disposition</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.StringClientParam DISPOSITION = new ca.uhn.fhir.rest.gclient.StringClientParam(
      SP_DISPOSITION);

  /**
   * Search parameter: <b>encounter</b>
   * <p>
   * Description: <b>Encounters associated with a billed line item</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.encounter</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "encounter", path = "ExplanationOfBenefit.item.encounter", description = "Encounters associated with a billed line item", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Encounter") }, target = {
          Encounter.class })
  public static final String SP_ENCOUNTER = "encounter";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>encounter</b>
   * <p>
   * Description: <b>Encounters associated with a billed line item</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.encounter</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam ENCOUNTER = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_ENCOUNTER);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:encounter</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_ENCOUNTER = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:encounter").toLocked();

  /**
   * Search parameter: <b>enterer</b>
   * <p>
   * Description: <b>The party responsible for the entry of the Claim</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.enterer</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "enterer", path = "ExplanationOfBenefit.enterer", description = "The party responsible for the entry of the Claim", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Practitioner") }, target = {
          Practitioner.class, PractitionerRole.class })
  public static final String SP_ENTERER = "enterer";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>enterer</b>
   * <p>
   * Description: <b>The party responsible for the entry of the Claim</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.enterer</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam ENTERER = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_ENTERER);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:enterer</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_ENTERER = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:enterer").toLocked();

  /**
   * Search parameter: <b>facility</b>
   * <p>
   * Description: <b>Facility responsible for the goods and services</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.facility</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "facility", path = "ExplanationOfBenefit.facility", description = "Facility responsible for the goods and services", type = "reference", target = {
      Location.class })
  public static final String SP_FACILITY = "facility";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>facility</b>
   * <p>
   * Description: <b>Facility responsible for the goods and services</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.facility</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam FACILITY = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_FACILITY);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:facility</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_FACILITY = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:facility").toLocked();

  /**
   * Search parameter: <b>identifier</b>
   * <p>
   * Description: <b>The business identifier of the Explanation of Benefit</b><br>
   * Type: <b>token</b><br>
   * Path: <b>ExplanationOfBenefit.identifier</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "identifier", path = "ExplanationOfBenefit.identifier", description = "The business identifier of the Explanation of Benefit", type = "token")
  public static final String SP_IDENTIFIER = "identifier";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>identifier</b>
   * <p>
   * Description: <b>The business identifier of the Explanation of Benefit</b><br>
   * Type: <b>token</b><br>
   * Path: <b>ExplanationOfBenefit.identifier</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.TokenClientParam IDENTIFIER = new ca.uhn.fhir.rest.gclient.TokenClientParam(
      SP_IDENTIFIER);

  /**
   * Search parameter: <b>item-udi</b>
   * <p>
   * Description: <b>UDI associated with a line item product or service</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.udi</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "item-udi", path = "ExplanationOfBenefit.item.udi", description = "UDI associated with a line item product or service", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Device") }, target = {
          Device.class })
  public static final String SP_ITEM_UDI = "item-udi";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>item-udi</b>
   * <p>
   * Description: <b>UDI associated with a line item product or service</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.udi</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam ITEM_UDI = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_ITEM_UDI);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:item-udi</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_ITEM_UDI = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:item-udi").toLocked();

  /**
   * Search parameter: <b>patient</b>
   * <p>
   * Description: <b>The reference to the patient</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.patient</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "patient", path = "ExplanationOfBenefit.patient", description = "The reference to the patient", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Patient") }, target = {
          Patient.class })
  public static final String SP_PATIENT = "patient";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>patient</b>
   * <p>
   * Description: <b>The reference to the patient</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.patient</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam PATIENT = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_PATIENT);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:patient</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_PATIENT = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:patient").toLocked();

  /**
   * Search parameter: <b>payee</b>
   * <p>
   * Description: <b>The party receiving any payment for the Claim</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.payee.party</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "payee", path = "ExplanationOfBenefit.payee.party", description = "The party receiving any payment for the Claim", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Patient"),
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Practitioner"),
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for RelatedPerson") }, target = {
          Organization.class, Patient.class, Practitioner.class, PractitionerRole.class, RelatedPerson.class })
  public static final String SP_PAYEE = "payee";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>payee</b>
   * <p>
   * Description: <b>The party receiving any payment for the Claim</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.payee.party</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam PAYEE = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_PAYEE);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:payee</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_PAYEE = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:payee").toLocked();

  /**
   * Search parameter: <b>procedure-udi</b>
   * <p>
   * Description: <b>UDI associated with a procedure</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.procedure.udi</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "procedure-udi", path = "ExplanationOfBenefit.procedure.udi", description = "UDI associated with a procedure", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Device") }, target = {
          Device.class })
  public static final String SP_PROCEDURE_UDI = "procedure-udi";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>procedure-udi</b>
   * <p>
   * Description: <b>UDI associated with a procedure</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.procedure.udi</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam PROCEDURE_UDI = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_PROCEDURE_UDI);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:procedure-udi</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_PROCEDURE_UDI = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:procedure-udi").toLocked();

  /**
   * Search parameter: <b>provider</b>
   * <p>
   * Description: <b>The reference to the provider</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.provider</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "provider", path = "ExplanationOfBenefit.provider", description = "The reference to the provider", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Practitioner") }, target = {
          Organization.class, Practitioner.class, PractitionerRole.class })
  public static final String SP_PROVIDER = "provider";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>provider</b>
   * <p>
   * Description: <b>The reference to the provider</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.provider</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam PROVIDER = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_PROVIDER);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:provider</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_PROVIDER = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:provider").toLocked();

  /**
   * Search parameter: <b>status</b>
   * <p>
   * Description: <b>Status of the instance</b><br>
   * Type: <b>token</b><br>
   * Path: <b>ExplanationOfBenefit.status</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "status", path = "ExplanationOfBenefit.status", description = "Status of the instance", type = "token")
  public static final String SP_STATUS = "status";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>status</b>
   * <p>
   * Description: <b>Status of the instance</b><br>
   * Type: <b>token</b><br>
   * Path: <b>ExplanationOfBenefit.status</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.TokenClientParam STATUS = new ca.uhn.fhir.rest.gclient.TokenClientParam(
      SP_STATUS);

  /**
   * Search parameter: <b>subdetail-udi</b>
   * <p>
   * Description: <b>UDI associated with a line item detail subdetail product or
   * service</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.detail.subDetail.udi</b><br>
   * </p>
   */
  @SearchParamDefinition(name = "subdetail-udi", path = "ExplanationOfBenefit.item.detail.subDetail.udi", description = "UDI associated with a line item detail subdetail product or service", type = "reference", providesMembershipIn = {
      @ca.uhn.fhir.model.api.annotation.Compartment(name = "Base FHIR compartment definition for Device") }, target = {
          Device.class })
  public static final String SP_SUBDETAIL_UDI = "subdetail-udi";
  /**
   * <b>Fluent Client</b> search parameter constant for <b>subdetail-udi</b>
   * <p>
   * Description: <b>UDI associated with a line item detail subdetail product or
   * service</b><br>
   * Type: <b>reference</b><br>
   * Path: <b>ExplanationOfBenefit.item.detail.subDetail.udi</b><br>
   * </p>
   */
  public static final ca.uhn.fhir.rest.gclient.ReferenceClientParam SUBDETAIL_UDI = new ca.uhn.fhir.rest.gclient.ReferenceClientParam(
      SP_SUBDETAIL_UDI);

  /**
   * Constant for fluent queries to be used to add include statements. Specifies
   * the path value of "<b>ExplanationOfBenefit:subdetail-udi</b>".
   */
  public static final ca.uhn.fhir.model.api.Include INCLUDE_SUBDETAIL_UDI = new ca.uhn.fhir.model.api.Include(
      "ExplanationOfBenefit:subdetail-udi").toLocked();

}
